blob: db08eab295d3d28245db75623c64979f36f43318 [file] [log] [blame]
Benjamin Peterson90f5ba52010-03-11 22:53:45 +00001#!/usr/bin/env python3
Marc-André Lemburg246d8472003-04-24 11:36:11 +00002
Brett Cannon8ab27df2003-08-05 03:52:04 +00003""" This module tries to retrieve as much platform-identifying data as
Marc-André Lemburg246d8472003-04-24 11:36:11 +00004 possible. It makes this information available via function APIs.
5
6 If called from the command line, it prints the platform
7 information concatenated as single string to stdout. The output
8 format is useable as part of a filename.
9
10"""
11# This module is maintained by Marc-Andre Lemburg <mal@egenix.com>.
12# If you find problems, please submit bug reports/patches via the
Benjamin Peterson4ac9ce42009-10-04 14:49:41 +000013# Python bug tracker (http://bugs.python.org) and assign them to "lemburg".
Marc-André Lemburg246d8472003-04-24 11:36:11 +000014#
Marc-André Lemburg246d8472003-04-24 11:36:11 +000015# Still needed:
16# * more support for WinCE
17# * support for MS-DOS (PythonDX ?)
18# * support for Amiga and other still unsupported platforms running Python
19# * support for additional Linux distributions
20#
Brett Cannon8ab27df2003-08-05 03:52:04 +000021# Many thanks to all those who helped adding platform-specific
Marc-André Lemburg246d8472003-04-24 11:36:11 +000022# checks (in no particular order):
23#
24# Charles G Waldman, David Arnold, Gordon McMillan, Ben Darnell,
25# Jeff Bauer, Cliff Crawford, Ivan Van Laningham, Josef
26# Betancourt, Randall Hopper, Karl Putland, John Farrell, Greg
27# Andruk, Just van Rossum, Thomas Heller, Mark R. Levinson, Mark
28# Hammond, Bill Tutt, Hans Nowak, Uwe Zessin (OpenVMS support),
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +000029# Colin Kong, Trent Mick, Guido van Rossum, Anthony Baxter
Marc-André Lemburg246d8472003-04-24 11:36:11 +000030#
31# History:
Marc-André Lemburg380f4172005-11-07 16:11:02 +000032#
33# <see CVS and SVN checkin messages for history>
34#
Alexandre Vassalottie52e3782009-07-17 09:18:18 +000035# 1.0.7 - added DEV_NULL
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +000036# 1.0.6 - added linux_distribution()
37# 1.0.5 - fixed Java support to allow running the module on Jython
38# 1.0.4 - added IronPython support
Marc-André Lemburgcdc79232004-06-19 17:17:00 +000039# 1.0.3 - added normalization of Windows system name
Marc-André Lemburg91e83e22004-03-25 18:35:12 +000040# 1.0.2 - added more Windows support
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +000041# 1.0.1 - reformatted to make doc.py happy
Marc-André Lemburg246d8472003-04-24 11:36:11 +000042# 1.0.0 - reformatted a bit and checked into Python CVS
43# 0.8.0 - added sys.version parser and various new access
44# APIs (python_version(), python_compiler(), etc.)
45# 0.7.2 - fixed architecture() to use sizeof(pointer) where available
46# 0.7.1 - added support for Caldera OpenLinux
47# 0.7.0 - some fixes for WinCE; untabified the source file
48# 0.6.2 - support for OpenVMS - requires version 1.5.2-V006 or higher and
49# vms_lib.getsyi() configured
50# 0.6.1 - added code to prevent 'uname -p' on platforms which are
51# known not to support it
52# 0.6.0 - fixed win32_ver() to hopefully work on Win95,98,NT and Win2k;
53# did some cleanup of the interfaces - some APIs have changed
54# 0.5.5 - fixed another type in the MacOS code... should have
55# used more coffee today ;-)
56# 0.5.4 - fixed a few typos in the MacOS code
57# 0.5.3 - added experimental MacOS support; added better popen()
58# workarounds in _syscmd_ver() -- still not 100% elegant
59# though
60# 0.5.2 - fixed uname() to return '' instead of 'unknown' in all
61# return values (the system uname command tends to return
62# 'unknown' instead of just leaving the field emtpy)
63# 0.5.1 - included code for slackware dist; added exception handlers
64# to cover up situations where platforms don't have os.popen
65# (e.g. Mac) or fail on socket.gethostname(); fixed libc
66# detection RE
67# 0.5.0 - changed the API names referring to system commands to *syscmd*;
68# added java_ver(); made syscmd_ver() a private
69# API (was system_ver() in previous versions) -- use uname()
70# instead; extended the win32_ver() to also return processor
71# type information
72# 0.4.0 - added win32_ver() and modified the platform() output for WinXX
73# 0.3.4 - fixed a bug in _follow_symlinks()
74# 0.3.3 - fixed popen() and "file" command invokation bugs
75# 0.3.2 - added architecture() API and support for it in platform()
76# 0.3.1 - fixed syscmd_ver() RE to support Windows NT
77# 0.3.0 - added system alias support
78# 0.2.3 - removed 'wince' again... oh well.
79# 0.2.2 - added 'wince' to syscmd_ver() supported platforms
80# 0.2.1 - added cache logic and changed the platform string format
81# 0.2.0 - changed the API to use functions instead of module globals
82# since some action take too long to be run on module import
83# 0.1.0 - first release
84#
85# You can always get the latest version of this module at:
86#
87# http://www.egenix.com/files/python/platform.py
88#
89# If that URL should fail, try contacting the author.
90
91__copyright__ = """
92 Copyright (c) 1999-2000, Marc-Andre Lemburg; mailto:mal@lemburg.com
Benjamin Peterson46a99002010-01-09 18:45:30 +000093 Copyright (c) 2000-2010, eGenix.com Software GmbH; mailto:info@egenix.com
Marc-André Lemburg246d8472003-04-24 11:36:11 +000094
95 Permission to use, copy, modify, and distribute this software and its
96 documentation for any purpose and without fee or royalty is hereby granted,
97 provided that the above copyright notice appear in all copies and that
98 both that copyright notice and this permission notice appear in
99 supporting documentation or portions thereof, including modifications,
100 that you make.
101
102 EGENIX.COM SOFTWARE GMBH DISCLAIMS ALL WARRANTIES WITH REGARD TO
103 THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
104 FITNESS, IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL,
105 INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
106 FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
107 NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
108 WITH THE USE OR PERFORMANCE OF THIS SOFTWARE !
109
110"""
111
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000112__version__ = '1.0.7'
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000113
Larry Hastings68386bc2012-06-24 14:30:41 -0700114import collections
Jesus Ceafc990e92012-10-04 13:51:43 +0200115import sys, os, re, subprocess
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000116
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000117### Globals & Constants
118
119# Determine the platform's /dev/null device
120try:
121 DEV_NULL = os.devnull
122except AttributeError:
123 # os.devnull was added in Python 2.4, so emulate it for earlier
124 # Python versions
Jesus Cea4791a242012-10-05 03:15:39 +0200125 if sys.platform in ('dos','win32','win16'):
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000126 # Use the old CP/M NUL as device name
127 DEV_NULL = 'NUL'
128 else:
129 # Standard Unix uses /dev/null
130 DEV_NULL = '/dev/null'
131
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000132### Platform specific APIs
133
Antoine Pitrouba7c2262011-10-07 13:26:59 +0200134_libc_search = re.compile(b'(__libc_init)'
135 b'|'
136 b'(GLIBC_([0-9.]+))'
137 b'|'
138 br'(libc(_\w+)?\.so(?:\.(\d[0-9.]*))?)', re.ASCII)
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +0000139
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000140def libc_ver(executable=sys.executable,lib='',version='',
141
Antoine Pitrouba7c2262011-10-07 13:26:59 +0200142 chunksize=16384):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000143
Brett Cannon8ab27df2003-08-05 03:52:04 +0000144 """ Tries to determine the libc version that the file executable
145 (which defaults to the Python interpreter) is linked against.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000146
147 Returns a tuple of strings (lib,version) which default to the
148 given parameters in case the lookup fails.
149
150 Note that the function has intimate knowledge of how different
Brett Cannon8ab27df2003-08-05 03:52:04 +0000151 libc versions add symbols to the executable and thus is probably
152 only useable for executables compiled using gcc.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000153
154 The file is read and scanned in chunks of chunksize bytes.
155
156 """
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000157 if hasattr(os.path, 'realpath'):
158 # Python 2.2 introduced os.path.realpath(); it is used
159 # here to work around problems with Cygwin not being
160 # able to open symlinks for reading
161 executable = os.path.realpath(executable)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000162 f = open(executable,'rb')
Antoine Pitrouba7c2262011-10-07 13:26:59 +0200163 binary = f.read(chunksize)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000164 pos = 0
165 while 1:
Antoine Pitrouba7c2262011-10-07 13:26:59 +0200166 if b'libc' in binary or b'GLIBC' in binary:
167 m = _libc_search.search(binary,pos)
168 else:
169 m = None
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000170 if not m:
Antoine Pitrouba7c2262011-10-07 13:26:59 +0200171 binary = f.read(chunksize)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000172 if not binary:
173 break
174 pos = 0
175 continue
Antoine Pitrouba7c2262011-10-07 13:26:59 +0200176 libcinit,glibc,glibcversion,so,threads,soversion = [
177 s.decode('latin1') if s is not None else s
178 for s in m.groups()]
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000179 if libcinit and not lib:
180 lib = 'libc'
181 elif glibc:
182 if lib != 'glibc':
183 lib = 'glibc'
184 version = glibcversion
185 elif glibcversion > version:
186 version = glibcversion
187 elif so:
188 if lib != 'glibc':
189 lib = 'libc'
Victor Stinner87448812011-12-15 21:42:03 +0100190 if soversion and soversion > version:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000191 version = soversion
192 if threads and version[-len(threads):] != threads:
193 version = version + threads
194 pos = m.end()
195 f.close()
196 return lib,version
197
198def _dist_try_harder(distname,version,id):
199
Tim Peters0eadaac2003-04-24 16:02:54 +0000200 """ Tries some special tricks to get the distribution
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000201 information in case the default method fails.
202
203 Currently supports older SuSE Linux, Caldera OpenLinux and
204 Slackware Linux distributions.
205
206 """
207 if os.path.exists('/var/adm/inst-log/info'):
208 # SuSE Linux stores distribution information in that file
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000209 distname = 'SuSE'
Florent Xicluna7dde7922010-09-03 19:52:03 +0000210 for line in open('/var/adm/inst-log/info'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000211 tv = line.split()
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000212 if len(tv) == 2:
213 tag,value = tv
214 else:
215 continue
216 if tag == 'MIN_DIST_VERSION':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000217 version = value.strip()
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000218 elif tag == 'DIST_IDENT':
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000219 values = value.split('-')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000220 id = values[2]
221 return distname,version,id
222
223 if os.path.exists('/etc/.installed'):
224 # Caldera OpenLinux has some infos in that file (thanks to Colin Kong)
Florent Xicluna7dde7922010-09-03 19:52:03 +0000225 for line in open('/etc/.installed'):
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000226 pkg = line.split('-')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000227 if len(pkg) >= 2 and pkg[0] == 'OpenLinux':
228 # XXX does Caldera support non Intel platforms ? If yes,
229 # where can we find the needed id ?
230 return 'OpenLinux',pkg[1],id
231
232 if os.path.isdir('/usr/lib/setup'):
233 # Check for slackware verson tag file (thanks to Greg Andruk)
234 verfiles = os.listdir('/usr/lib/setup')
Guido van Rossum843c7342004-05-04 18:18:59 +0000235 for n in range(len(verfiles)-1, -1, -1):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000236 if verfiles[n][:14] != 'slack-version-':
237 del verfiles[n]
238 if verfiles:
239 verfiles.sort()
240 distname = 'slackware'
241 version = verfiles[-1][14:]
242 return distname,version,id
243
244 return distname,version,id
245
Antoine Pitroufd036452008-08-19 17:56:33 +0000246_release_filename = re.compile(r'(\w+)[-_](release|version)', re.ASCII)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000247_lsb_release_version = re.compile(r'(.+)'
248 ' release '
249 '([\d.]+)'
Antoine Pitroufd036452008-08-19 17:56:33 +0000250 '[^(]*(?:\((.+)\))?', re.ASCII)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000251_release_version = re.compile(r'([^0-9]+)'
252 '(?: release )?'
253 '([\d.]+)'
Antoine Pitroufd036452008-08-19 17:56:33 +0000254 '[^(]*(?:\((.+)\))?', re.ASCII)
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +0000255
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000256# See also http://www.novell.com/coolsolutions/feature/11251.html
Thomas Wouters9fe394c2007-02-05 01:24:16 +0000257# and http://linuxmafia.com/faq/Admin/release-files.html
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000258# and http://data.linux-ntfs.org/rpm/whichrpm
259# and http://www.die.net/doc/linux/man/man1/lsb_release.1.html
Marc-André Lemburg380f4172005-11-07 16:11:02 +0000260
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000261_supported_dists = (
262 'SuSE', 'debian', 'fedora', 'redhat', 'centos',
263 'mandrake', 'mandriva', 'rocks', 'slackware', 'yellowdog', 'gentoo',
Antoine Pitrouab888032012-06-24 22:20:18 +0200264 'UnitedLinux', 'turbolinux', 'arch', 'mageia')
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000265
266def _parse_release_file(firstline):
Thomas Wouters9fe394c2007-02-05 01:24:16 +0000267
Benjamin Peterson25001472010-01-25 03:37:42 +0000268 # Default to empty 'version' and 'id' strings. Both defaults are used
269 # when 'firstline' is empty. 'id' defaults to empty when an id can not
270 # be deduced.
271 version = ''
272 id = ''
273
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000274 # Parse the first line
275 m = _lsb_release_version.match(firstline)
276 if m is not None:
277 # LSB format: "distro release x.x (codename)"
278 return tuple(m.groups())
279
280 # Pre-LSB format: "distro x.x (codename)"
281 m = _release_version.match(firstline)
282 if m is not None:
283 return tuple(m.groups())
284
285 # Unkown format... take the first two words
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000286 l = firstline.strip().split()
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000287 if l:
288 version = l[0]
289 if len(l) > 1:
290 id = l[1]
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000291 return '', version, id
292
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000293def linux_distribution(distname='', version='', id='',
294
295 supported_dists=_supported_dists,
296 full_distribution_name=1):
297
298 """ Tries to determine the name of the Linux OS distribution name.
299
300 The function first looks for a distribution release file in
301 /etc and then reverts to _dist_try_harder() in case no
302 suitable files are found.
303
304 supported_dists may be given to define the set of Linux
305 distributions to look for. It defaults to a list of currently
306 supported Linux distributions identified by their release file
307 name.
308
309 If full_distribution_name is true (default), the full
310 distribution read from the OS is returned. Otherwise the short
311 name taken from supported_dists is used.
312
313 Returns a tuple (distname,version,id) which default to the
314 args given as parameters.
315
316 """
317 try:
318 etc = os.listdir('/etc')
319 except os.error:
320 # Probably not a Unix system
321 return distname,version,id
322 etc.sort()
323 for file in etc:
324 m = _release_filename.match(file)
325 if m is not None:
326 _distname,dummy = m.groups()
327 if _distname in supported_dists:
328 distname = _distname
329 break
330 else:
331 return _dist_try_harder(distname,version,id)
Thomas Wouters9fe394c2007-02-05 01:24:16 +0000332
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000333 # Read the first line
Florent Xicluna7dde7922010-09-03 19:52:03 +0000334 with open('/etc/'+file, 'r') as f:
335 firstline = f.readline()
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000336 _distname, _version, _id = _parse_release_file(firstline)
337
338 if _distname and full_distribution_name:
339 distname = _distname
340 if _version:
341 version = _version
342 if _id:
343 id = _id
344 return distname, version, id
345
346# To maintain backwards compatibility:
Thomas Wouters9fe394c2007-02-05 01:24:16 +0000347
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000348def dist(distname='',version='',id='',
349
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000350 supported_dists=_supported_dists):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000351
Brett Cannon8ab27df2003-08-05 03:52:04 +0000352 """ Tries to determine the name of the Linux OS distribution name.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000353
354 The function first looks for a distribution release file in
355 /etc and then reverts to _dist_try_harder() in case no
356 suitable files are found.
357
Brett Cannon8ab27df2003-08-05 03:52:04 +0000358 Returns a tuple (distname,version,id) which default to the
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000359 args given as parameters.
360
361 """
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000362 return linux_distribution(distname, version, id,
363 supported_dists=supported_dists,
364 full_distribution_name=0)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000365
Antoine Pitrou877766d2011-03-19 17:00:37 +0100366def popen(cmd, mode='r', bufsize=-1):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000367
368 """ Portable popen() interface.
369 """
Victor Stinner25000d42011-05-24 00:16:16 +0200370 import warnings
371 warnings.warn('use os.popen instead', DeprecationWarning, stacklevel=2)
Victor Stinner1dfd3802011-03-03 12:54:07 +0000372 return os.popen(cmd, mode, bufsize)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000373
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000374def _norm_version(version, build=''):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000375
Brett Cannon8ab27df2003-08-05 03:52:04 +0000376 """ Normalize the version and build strings and return a single
Walter Dörwalde5a7fad2005-11-21 17:01:27 +0000377 version string using the format major.minor.build (or patchlevel).
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000378 """
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000379 l = version.split('.')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000380 if build:
381 l.append(build)
382 try:
383 ints = map(int,l)
384 except ValueError:
385 strings = l
386 else:
Neal Norwitz78a70bd2007-08-30 06:16:26 +0000387 strings = list(map(str,ints))
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000388 version = '.'.join(strings[:3])
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000389 return version
390
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +0000391_ver_output = re.compile(r'(?:([\w ]+) ([\w.]+) '
392 '.*'
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000393 '\[.* ([\d.]+)\])')
394
395# Examples of VER command output:
396#
397# Windows 2000: Microsoft Windows 2000 [Version 5.00.2195]
398# Windows XP: Microsoft Windows XP [Version 5.1.2600]
399# Windows Vista: Microsoft Windows [Version 6.0.6002]
400#
401# Note that the "Version" string gets localized on different
402# Windows versions.
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +0000403
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000404def _syscmd_ver(system='', release='', version='',
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000405
Jesus Ceaf1af7052012-10-05 02:48:46 +0200406 supported_platforms=('win32','win16','dos')):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000407
408 """ Tries to figure out the OS version used and returns
409 a tuple (system,release,version).
Tim Peters0eadaac2003-04-24 16:02:54 +0000410
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000411 It uses the "ver" shell command for this which is known
Jesus Ceaf1af7052012-10-05 02:48:46 +0200412 to exists on Windows, DOS. XXX Others too ?
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000413
414 In case this fails, the given parameters are used as
415 defaults.
416
417 """
418 if sys.platform not in supported_platforms:
419 return system,release,version
420
421 # Try some common cmd strings
422 for cmd in ('ver','command /c ver','cmd /c ver'):
423 try:
424 pipe = popen(cmd)
425 info = pipe.read()
426 if pipe.close():
Collin Winterce36ad82007-08-30 01:19:48 +0000427 raise os.error('command failed')
Ezio Melotti13925002011-03-16 11:05:33 +0200428 # XXX How can I suppress shell errors from being written
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000429 # to stderr ?
Guido van Rossumb940e112007-01-10 16:19:56 +0000430 except os.error as why:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000431 #print 'Command %s failed: %s' % (cmd,why)
432 continue
Guido van Rossumb940e112007-01-10 16:19:56 +0000433 except IOError as why:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000434 #print 'Command %s failed: %s' % (cmd,why)
435 continue
436 else:
437 break
438 else:
439 return system,release,version
440
441 # Parse the output
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000442 info = info.strip()
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +0000443 m = _ver_output.match(info)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000444 if m is not None:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000445 system,release,version = m.groups()
446 # Strip trailing dots from version and release
447 if release[-1] == '.':
448 release = release[:-1]
449 if version[-1] == '.':
450 version = version[:-1]
451 # Normalize the version and build strings (eliminating additional
452 # zeros)
453 version = _norm_version(version)
454 return system,release,version
455
456def _win32_getvalue(key,name,default=''):
457
458 """ Read a value for name from the registry key.
459
460 In case this fails, default is returned.
461
462 """
Christian Heimes02781dc2008-03-21 01:11:52 +0000463 try:
464 # Use win32api if available
465 from win32api import RegQueryValueEx
466 except ImportError:
Georg Brandl38feaf02008-05-25 07:45:51 +0000467 # On Python 2.0 and later, emulate using winreg
468 import winreg
469 RegQueryValueEx = winreg.QueryValueEx
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000470 try:
471 return RegQueryValueEx(key,name)
472 except:
473 return default
474
475def win32_ver(release='',version='',csd='',ptype=''):
476
477 """ Get additional version information from the Windows Registry
478 and return a tuple (version,csd,ptype) referring to version
Brian Curtin10dda6e2012-02-01 15:14:00 -0600479 number, CSD level (service pack), and OS type (multi/single
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000480 processor).
481
482 As a hint: ptype returns 'Uniprocessor Free' on single
483 processor NT machines and 'Multiprocessor Free' on multi
484 processor machines. The 'Free' refers to the OS version being
485 free of debugging code. It could also state 'Checked' which
486 means the OS version uses debugging code, i.e. code that
487 checks arguments, ranges, etc. (Thomas Heller).
488
Christian Heimes02781dc2008-03-21 01:11:52 +0000489 Note: this function works best with Mark Hammond's win32
490 package installed, but also on Python 2.3 and later. It
491 obviously only runs on Win32 compatible platforms.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000492
493 """
494 # XXX Is there any way to find out the processor type on WinXX ?
495 # XXX Is win32 available on Windows CE ?
Marc-André Lemburg91e83e22004-03-25 18:35:12 +0000496 #
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000497 # Adapted from code posted by Karl Putland to comp.lang.python.
Marc-André Lemburg91e83e22004-03-25 18:35:12 +0000498 #
499 # The mappings between reg. values and release names can be found
500 # here: http://msdn.microsoft.com/library/en-us/sysinfo/base/osversioninfo_str.asp
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000501
502 # Import the needed APIs
503 try:
504 import win32api
Christian Heimes02781dc2008-03-21 01:11:52 +0000505 from win32api import RegQueryValueEx, RegOpenKeyEx, \
506 RegCloseKey, GetVersionEx
507 from win32con import HKEY_LOCAL_MACHINE, VER_PLATFORM_WIN32_NT, \
508 VER_PLATFORM_WIN32_WINDOWS, VER_NT_WORKSTATION
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000509 except ImportError:
Christian Heimes02781dc2008-03-21 01:11:52 +0000510 # Emulate the win32api module using Python APIs
511 try:
512 sys.getwindowsversion
513 except AttributeError:
514 # No emulation possible, so return the defaults...
515 return release,version,csd,ptype
516 else:
Georg Brandl38feaf02008-05-25 07:45:51 +0000517 # Emulation using winreg (added in Python 2.0) and
Christian Heimes02781dc2008-03-21 01:11:52 +0000518 # sys.getwindowsversion() (added in Python 2.3)
Georg Brandl38feaf02008-05-25 07:45:51 +0000519 import winreg
Christian Heimes02781dc2008-03-21 01:11:52 +0000520 GetVersionEx = sys.getwindowsversion
Georg Brandl38feaf02008-05-25 07:45:51 +0000521 RegQueryValueEx = winreg.QueryValueEx
522 RegOpenKeyEx = winreg.OpenKeyEx
523 RegCloseKey = winreg.CloseKey
524 HKEY_LOCAL_MACHINE = winreg.HKEY_LOCAL_MACHINE
Christian Heimes02781dc2008-03-21 01:11:52 +0000525 VER_PLATFORM_WIN32_WINDOWS = 1
526 VER_PLATFORM_WIN32_NT = 2
527 VER_NT_WORKSTATION = 1
Brian Curtin6e2824d2010-05-06 03:05:50 +0000528 VER_NT_SERVER = 3
529 REG_SZ = 1
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000530
531 # Find out the registry key and some general version infos
Brian Curtin6e2824d2010-05-06 03:05:50 +0000532 winver = GetVersionEx()
533 maj,min,buildno,plat,csd = winver
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000534 version = '%i.%i.%i' % (maj,min,buildno & 0xFFFF)
Brian Curtin6e2824d2010-05-06 03:05:50 +0000535 if hasattr(winver, "service_pack"):
536 if winver.service_pack != "":
537 csd = 'SP%s' % winver.service_pack_major
538 else:
539 if csd[:13] == 'Service Pack ':
540 csd = 'SP' + csd[13:]
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000541
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000542 if plat == VER_PLATFORM_WIN32_WINDOWS:
543 regkey = 'SOFTWARE\\Microsoft\\Windows\\CurrentVersion'
544 # Try to guess the release name
545 if maj == 4:
546 if min == 0:
547 release = '95'
Marc-André Lemburg91e83e22004-03-25 18:35:12 +0000548 elif min == 10:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000549 release = '98'
Marc-André Lemburg91e83e22004-03-25 18:35:12 +0000550 elif min == 90:
551 release = 'Me'
552 else:
553 release = 'postMe'
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000554 elif maj == 5:
555 release = '2000'
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000556
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000557 elif plat == VER_PLATFORM_WIN32_NT:
558 regkey = 'SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion'
559 if maj <= 4:
560 release = 'NT'
561 elif maj == 5:
Marc-André Lemburg91e83e22004-03-25 18:35:12 +0000562 if min == 0:
563 release = '2000'
564 elif min == 1:
565 release = 'XP'
566 elif min == 2:
567 release = '2003Server'
568 else:
569 release = 'post2003'
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000570 elif maj == 6:
Brian Curtin6e2824d2010-05-06 03:05:50 +0000571 if hasattr(winver, "product_type"):
572 product_type = winver.product_type
573 else:
574 product_type = VER_NT_WORKSTATION
575 # Without an OSVERSIONINFOEX capable sys.getwindowsversion(),
576 # or help from the registry, we cannot properly identify
577 # non-workstation versions.
Christian Heimes02781dc2008-03-21 01:11:52 +0000578 try:
Brian Curtin6e2824d2010-05-06 03:05:50 +0000579 key = RegOpenKeyEx(HKEY_LOCAL_MACHINE, regkey)
580 name, type = RegQueryValueEx(key, "ProductName")
581 # Discard any type that isn't REG_SZ
582 if type == REG_SZ and name.find("Server") != -1:
583 product_type = VER_NT_SERVER
584 except WindowsError:
585 # Use default of VER_NT_WORKSTATION
586 pass
587
588 if min == 0:
589 if product_type == VER_NT_WORKSTATION:
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000590 release = 'Vista'
591 else:
Brian Curtin6e2824d2010-05-06 03:05:50 +0000592 release = '2008Server'
593 elif min == 1:
594 if product_type == VER_NT_WORKSTATION:
595 release = '7'
596 else:
597 release = '2008ServerR2'
Brian Curtin0b960f52012-10-11 16:07:52 -0500598 elif min == 2:
599 if product_type == VER_NT_WORKSTATION:
600 release = '8'
601 else:
602 release = '2012Server'
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000603 else:
Brian Curtin0b960f52012-10-11 16:07:52 -0500604 release = 'post2012Server'
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000605
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000606 else:
607 if not release:
608 # E.g. Win3.1 with win32s
609 release = '%i.%i' % (maj,min)
610 return release,version,csd,ptype
611
612 # Open the registry key
613 try:
Christian Heimes02781dc2008-03-21 01:11:52 +0000614 keyCurVer = RegOpenKeyEx(HKEY_LOCAL_MACHINE, regkey)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000615 # Get a value to make sure the key exists...
Christian Heimes02781dc2008-03-21 01:11:52 +0000616 RegQueryValueEx(keyCurVer, 'SystemRoot')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000617 except:
618 return release,version,csd,ptype
Tim Peters0eadaac2003-04-24 16:02:54 +0000619
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000620 # Parse values
621 #subversion = _win32_getvalue(keyCurVer,
622 # 'SubVersionNumber',
623 # ('',1))[0]
624 #if subversion:
625 # release = release + subversion # 95a, 95b, etc.
626 build = _win32_getvalue(keyCurVer,
627 'CurrentBuildNumber',
628 ('',1))[0]
629 ptype = _win32_getvalue(keyCurVer,
630 'CurrentType',
631 (ptype,1))[0]
632
633 # Normalize version
634 version = _norm_version(version,build)
635
636 # Close key
637 RegCloseKey(keyCurVer)
638 return release,version,csd,ptype
639
640def _mac_ver_lookup(selectors,default=None):
641
Benjamin Petersonebacd262008-05-29 21:09:51 +0000642 from _gestalt import gestalt
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000643 l = []
644 append = l.append
645 for selector in selectors:
646 try:
647 append(gestalt(selector))
Benjamin Petersonebacd262008-05-29 21:09:51 +0000648 except (RuntimeError, OSError):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000649 append(default)
650 return l
651
652def _bcd2str(bcd):
653
654 return hex(bcd)[2:]
655
Ronald Oussorene186e382010-07-23 11:54:59 +0000656def _mac_ver_gestalt():
657 """
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000658 Thanks to Mark R. Levinson for mailing documentation links and
659 code examples for this function. Documentation for the
660 gestalt() API is available online at:
661
662 http://www.rgaros.nl/gestalt/
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000663 """
664 # Check whether the version info module is available
665 try:
Benjamin Petersonebacd262008-05-29 21:09:51 +0000666 import _gestalt
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000667 except ImportError:
Ronald Oussorene186e382010-07-23 11:54:59 +0000668 return None
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000669 # Get the infos
Ronald Oussoren19258d52010-02-07 11:33:33 +0000670 sysv, sysa = _mac_ver_lookup(('sysv','sysa'))
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000671 # Decode the infos
672 if sysv:
673 major = (sysv & 0xFF00) >> 8
674 minor = (sysv & 0x00F0) >> 4
675 patch = (sysv & 0x000F)
Christian Heimese4ca8152008-05-08 17:18:53 +0000676
677 if (major, minor) >= (10, 4):
678 # the 'sysv' gestald cannot return patchlevels
679 # higher than 9. Apple introduced 3 new
680 # gestalt codes in 10.4 to deal with this
681 # issue (needed because patch levels can
682 # run higher than 9, such as 10.4.11)
683 major,minor,patch = _mac_ver_lookup(('sys1','sys2','sys3'))
684 release = '%i.%i.%i' %(major, minor, patch)
685 else:
686 release = '%s.%i.%i' % (_bcd2str(major),minor,patch)
Benjamin Peterson856ff5f2008-05-29 21:22:40 +0000687
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000688 if sysa:
Tim Peters0eadaac2003-04-24 16:02:54 +0000689 machine = {0x1: '68k',
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000690 0x2: 'PowerPC',
691 0xa: 'i386'}.get(sysa,'')
Ronald Oussorene186e382010-07-23 11:54:59 +0000692
Ned Deily58e33502011-07-13 15:07:04 -0700693 versioninfo=('', '', '')
Ronald Oussorene186e382010-07-23 11:54:59 +0000694 return release,versioninfo,machine
695
696def _mac_ver_xml():
697 fn = '/System/Library/CoreServices/SystemVersion.plist'
698 if not os.path.exists(fn):
699 return None
700
701 try:
702 import plistlib
703 except ImportError:
704 return None
705
706 pl = plistlib.readPlist(fn)
707 release = pl['ProductVersion']
708 versioninfo=('', '', '')
Larry Hastings605a62d2012-06-24 04:33:36 -0700709 machine = os.uname().machine
Ronald Oussorenfcd77012010-08-03 07:42:42 +0000710 if machine in ('ppc', 'Power Macintosh'):
Ronald Oussorene186e382010-07-23 11:54:59 +0000711 # for compatibility with the gestalt based code
712 machine = 'PowerPC'
713
714 return release,versioninfo,machine
715
716
717def mac_ver(release='',versioninfo=('','',''),machine=''):
718
719 """ Get MacOS version information and return it as tuple (release,
720 versioninfo, machine) with versioninfo being a tuple (version,
721 dev_stage, non_release_version).
722
723 Entries which cannot be determined are set to the paramter values
724 which default to ''. All tuple entries are strings.
725 """
726
727 # First try reading the information from an XML file which should
728 # always be present
729 info = _mac_ver_xml()
730 if info is not None:
731 return info
732
733 # If that doesn't work for some reason fall back to reading the
734 # information using gestalt calls.
735 info = _mac_ver_gestalt()
736 if info is not None:
737 return info
738
739 # If that also doesn't work return the default values
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000740 return release,versioninfo,machine
741
Neal Norwitz9b924c62003-06-29 04:17:45 +0000742def _java_getprop(name,default):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000743
744 from java.lang import System
745 try:
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000746 value = System.getProperty(name)
747 if value is None:
748 return default
749 return value
750 except AttributeError:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000751 return default
752
753def java_ver(release='',vendor='',vminfo=('','',''),osinfo=('','','')):
Tim Peters0eadaac2003-04-24 16:02:54 +0000754
Brett Cannon8ab27df2003-08-05 03:52:04 +0000755 """ Version interface for Jython.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000756
757 Returns a tuple (release,vendor,vminfo,osinfo) with vminfo being
758 a tuple (vm_name,vm_release,vm_vendor) and osinfo being a
759 tuple (os_name,os_version,os_arch).
760
761 Values which cannot be determined are set to the defaults
762 given as parameters (which all default to '').
763
764 """
765 # Import the needed APIs
766 try:
767 import java.lang
768 except ImportError:
769 return release,vendor,vminfo,osinfo
770
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000771 vendor = _java_getprop('java.vendor', vendor)
772 release = _java_getprop('java.version', release)
773 vm_name, vm_release, vm_vendor = vminfo
774 vm_name = _java_getprop('java.vm.name', vm_name)
775 vm_vendor = _java_getprop('java.vm.vendor', vm_vendor)
776 vm_release = _java_getprop('java.vm.version', vm_release)
777 vminfo = vm_name, vm_release, vm_vendor
778 os_name, os_version, os_arch = osinfo
779 os_arch = _java_getprop('java.os.arch', os_arch)
780 os_name = _java_getprop('java.os.name', os_name)
781 os_version = _java_getprop('java.os.version', os_version)
782 osinfo = os_name, os_version, os_arch
Tim Peters0eadaac2003-04-24 16:02:54 +0000783
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000784 return release, vendor, vminfo, osinfo
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000785
786### System name aliasing
787
788def system_alias(system,release,version):
789
790 """ Returns (system,release,version) aliased to common
791 marketing names used for some systems.
792
793 It also does some reordering of the information in some cases
794 where it would otherwise cause confusion.
795
796 """
797 if system == 'Rhapsody':
798 # Apple's BSD derivative
799 # XXX How can we determine the marketing release number ?
800 return 'MacOS X Server',system+release,version
801
802 elif system == 'SunOS':
803 # Sun's OS
804 if release < '5':
805 # These releases use the old name SunOS
806 return system,release,version
807 # Modify release (marketing release = SunOS release - 3)
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000808 l = release.split('.')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000809 if l:
810 try:
811 major = int(l[0])
812 except ValueError:
813 pass
814 else:
815 major = major - 3
816 l[0] = str(major)
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000817 release = '.'.join(l)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000818 if release < '6':
819 system = 'Solaris'
820 else:
821 # XXX Whatever the new SunOS marketing name is...
822 system = 'Solaris'
823
824 elif system == 'IRIX64':
825 # IRIX reports IRIX64 on platforms with 64-bit support; yet it
826 # is really a version and not a different platform, since 32-bit
827 # apps are also supported..
828 system = 'IRIX'
829 if version:
830 version = version + ' (64bit)'
831 else:
832 version = '64bit'
833
834 elif system in ('win32','win16'):
835 # In case one of the other tricks
836 system = 'Windows'
837
838 return system,release,version
839
840### Various internal helpers
841
842def _platform(*args):
843
844 """ Helper to format the platform string in a filename
845 compatible format e.g. "system-version-machine".
846 """
847 # Format the platform string
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000848 platform = '-'.join(x.strip() for x in filter(len, args))
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000849
850 # Cleanup some possible filename obstacles...
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000851 platform = platform.replace(' ','_')
852 platform = platform.replace('/','-')
853 platform = platform.replace('\\','-')
854 platform = platform.replace(':','-')
855 platform = platform.replace(';','-')
856 platform = platform.replace('"','-')
857 platform = platform.replace('(','-')
858 platform = platform.replace(')','-')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000859
860 # No need to report 'unknown' information...
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000861 platform = platform.replace('unknown','')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000862
863 # Fold '--'s and remove trailing '-'
864 while 1:
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000865 cleaned = platform.replace('--','-')
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000866 if cleaned == platform:
867 break
868 platform = cleaned
869 while platform[-1] == '-':
870 platform = platform[:-1]
871
872 return platform
873
874def _node(default=''):
875
876 """ Helper to determine the node name of this machine.
877 """
878 try:
879 import socket
880 except ImportError:
881 # No sockets...
882 return default
883 try:
884 return socket.gethostname()
885 except socket.error:
886 # Still not working...
887 return default
888
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000889def _follow_symlinks(filepath):
890
891 """ In case filepath is a symlink, follow it until a
892 real file is reached.
893 """
Georg Brandl673f7ef2008-01-05 21:20:19 +0000894 filepath = os.path.abspath(filepath)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000895 while os.path.islink(filepath):
896 filepath = os.path.normpath(
Hirokazu Yamamotob12716b2008-09-04 11:24:53 +0000897 os.path.join(os.path.dirname(filepath),os.readlink(filepath)))
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000898 return filepath
899
900def _syscmd_uname(option,default=''):
901
902 """ Interface to the system's uname command.
903 """
Jesus Cea4791a242012-10-05 03:15:39 +0200904 if sys.platform in ('dos','win32','win16'):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000905 # XXX Others too ?
906 return default
907 try:
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000908 f = os.popen('uname %s 2> %s' % (option, DEV_NULL))
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000909 except (AttributeError,os.error):
910 return default
Neal Norwitz9d72bb42007-04-17 08:48:32 +0000911 output = f.read().strip()
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000912 rc = f.close()
913 if not output or rc:
914 return default
915 else:
916 return output
917
918def _syscmd_file(target,default=''):
919
920 """ Interface to the system's file command.
921
922 The function uses the -b option of the file command to have it
Victor Stinnerddfb2c32010-08-13 16:30:15 +0000923 omit the filename in its output. Follow the symlinks. It returns
924 default in case the command should fail.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000925
926 """
Jesus Cea4791a242012-10-05 03:15:39 +0200927 if sys.platform in ('dos','win32','win16'):
Hirokazu Yamamotod26782e2008-09-01 14:35:47 +0000928 # XXX Others too ?
929 return default
Jesus Ceafc990e92012-10-04 13:51:43 +0200930 target = _follow_symlinks(target)
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000931 try:
Jesus Cea685fffa2012-10-05 05:21:42 +0200932 proc = subprocess.Popen(['file', target],
933 stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
Jesus Ceae8801e22012-10-04 13:56:23 +0200934
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000935 except (AttributeError,os.error):
936 return default
Jesus Ceacb959962012-10-05 05:31:31 +0200937 output = proc.communicate()[0].decode('latin-1')
Jesus Ceafc990e92012-10-04 13:51:43 +0200938 rc = proc.wait()
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000939 if not output or rc:
940 return default
941 else:
942 return output
943
944### Information about the used architecture
945
946# Default values for architecture; non-empty strings override the
947# defaults given as parameters
948_default_architecture = {
949 'win32': ('','WindowsPE'),
950 'win16': ('','Windows'),
951 'dos': ('','MSDOS'),
952}
953
Marc-André Lemburg366a0fe2003-04-24 11:46:35 +0000954def architecture(executable=sys.executable,bits='',linkage=''):
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000955
956 """ Queries the given executable (defaults to the Python interpreter
Brett Cannon8ab27df2003-08-05 03:52:04 +0000957 binary) for various architecture information.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000958
Brett Cannon8ab27df2003-08-05 03:52:04 +0000959 Returns a tuple (bits,linkage) which contains information about
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000960 the bit architecture and the linkage format used for the
961 executable. Both values are returned as strings.
962
963 Values that cannot be determined are returned as given by the
964 parameter presets. If bits is given as '', the sizeof(pointer)
965 (or sizeof(long) on Python version < 1.5.2) is used as
966 indicator for the supported pointer size.
967
968 The function relies on the system's "file" command to do the
969 actual work. This is available on most if not all Unix
Brett Cannon8ab27df2003-08-05 03:52:04 +0000970 platforms. On some non-Unix platforms where the "file" command
971 does not exist and the executable is set to the Python interpreter
972 binary defaults from _default_architecture are used.
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000973
974 """
975 # Use the sizeof(pointer) as default number of bits if nothing
976 # else is given as default.
977 if not bits:
978 import struct
979 try:
980 size = struct.calcsize('P')
981 except struct.error:
982 # Older installations can only query longs
983 size = struct.calcsize('l')
984 bits = str(size*8) + 'bit'
Tim Peters0eadaac2003-04-24 16:02:54 +0000985
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000986 # Get data from the 'file' system command
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000987 if executable:
Victor Stinnerddfb2c32010-08-13 16:30:15 +0000988 fileout = _syscmd_file(executable, '')
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000989 else:
Victor Stinnerddfb2c32010-08-13 16:30:15 +0000990 fileout = ''
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000991
Victor Stinnerddfb2c32010-08-13 16:30:15 +0000992 if not fileout and \
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000993 executable == sys.executable:
994 # "file" command did not return anything; we'll try to provide
Tim Peters0eadaac2003-04-24 16:02:54 +0000995 # some sensible defaults then...
Guido van Rossume2b70bc2006-08-18 22:13:04 +0000996 if sys.platform in _default_architecture:
Marc-André Lemburg246d8472003-04-24 11:36:11 +0000997 b,l = _default_architecture[sys.platform]
998 if b:
999 bits = b
1000 if l:
1001 linkage = l
1002 return bits,linkage
1003
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001004 if 'executable' not in fileout:
1005 # Format not supported
1006 return bits,linkage
1007
1008 # Bits
1009 if '32-bit' in fileout:
1010 bits = '32bit'
1011 elif 'N32' in fileout:
1012 # On Irix only
1013 bits = 'n32bit'
1014 elif '64-bit' in fileout:
1015 bits = '64bit'
1016
1017 # Linkage
1018 if 'ELF' in fileout:
1019 linkage = 'ELF'
1020 elif 'PE' in fileout:
1021 # E.g. Windows uses this format
1022 if 'Windows' in fileout:
1023 linkage = 'WindowsPE'
1024 else:
1025 linkage = 'PE'
1026 elif 'COFF' in fileout:
1027 linkage = 'COFF'
1028 elif 'MS-DOS' in fileout:
1029 linkage = 'MSDOS'
1030 else:
1031 # XXX the A.OUT format also falls under this class...
1032 pass
1033
1034 return bits,linkage
1035
1036### Portable uname() interface
Tim Peters0eadaac2003-04-24 16:02:54 +00001037
Larry Hastings68386bc2012-06-24 14:30:41 -07001038uname_result = collections.namedtuple("uname_result",
1039 "system node release version machine processor")
1040
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001041_uname_cache = None
1042
1043def uname():
1044
1045 """ Fairly portable uname interface. Returns a tuple
1046 of strings (system,node,release,version,machine,processor)
1047 identifying the underlying platform.
1048
1049 Note that unlike the os.uname function this also returns
Brett Cannon8ab27df2003-08-05 03:52:04 +00001050 possible processor information as an additional tuple entry.
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001051
1052 Entries which cannot be determined are set to ''.
1053
1054 """
1055 global _uname_cache
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001056 no_os_uname = 0
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001057
1058 if _uname_cache is not None:
1059 return _uname_cache
1060
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001061 processor = ''
1062
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001063 # Get some infos from the builtin os.uname API...
1064 try:
1065 system,node,release,version,machine = os.uname()
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001066 except AttributeError:
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001067 no_os_uname = 1
1068
Georg Brandl62e2ca22010-07-31 21:54:24 +00001069 if no_os_uname or not list(filter(None, (system, node, release, version, machine))):
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001070 # Hmm, no there is either no uname or uname has returned
1071 #'unknowns'... we'll have to poke around the system then.
1072 if no_os_uname:
1073 system = sys.platform
1074 release = ''
1075 version = ''
1076 node = _node()
1077 machine = ''
1078
Amaury Forgeot d'Arc7a019842008-06-17 21:42:46 +00001079 use_syscmd_ver = 1
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001080
1081 # Try win32_ver() on win32 platforms
1082 if system == 'win32':
1083 release,version,csd,ptype = win32_ver()
1084 if release and version:
1085 use_syscmd_ver = 0
Christian Heimes02781dc2008-03-21 01:11:52 +00001086 # Try to use the PROCESSOR_* environment variables
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001087 # available on Win XP and later; see
1088 # http://support.microsoft.com/kb/888731 and
1089 # http://www.geocities.com/rick_lively/MANUALS/ENV/MSWIN/PROCESSI.HTM
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001090 if not machine:
R. David Murrayca2edce2010-03-22 17:48:48 +00001091 # WOW64 processes mask the native architecture
1092 if "PROCESSOR_ARCHITEW6432" in os.environ:
1093 machine = os.environ.get("PROCESSOR_ARCHITEW6432", '')
1094 else:
1095 machine = os.environ.get('PROCESSOR_ARCHITECTURE', '')
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001096 if not processor:
1097 processor = os.environ.get('PROCESSOR_IDENTIFIER', machine)
Tim Peters0eadaac2003-04-24 16:02:54 +00001098
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001099 # Try the 'ver' system command available on some
1100 # platforms
1101 if use_syscmd_ver:
1102 system,release,version = _syscmd_ver(system)
Marc-André Lemburgcdc79232004-06-19 17:17:00 +00001103 # Normalize system to what win32_ver() normally returns
1104 # (_syscmd_ver() tends to return the vendor name as well)
1105 if system == 'Microsoft Windows':
1106 system = 'Windows'
Guido van Rossumcd16bf62007-06-13 18:07:49 +00001107 elif system == 'Microsoft' and release == 'Windows':
1108 # Under Windows Vista and Windows Server 2008,
1109 # Microsoft changed the output of the ver command. The
1110 # release is no longer printed. This causes the
1111 # system and release to be misidentified.
1112 system = 'Windows'
1113 if '6.0' == version[:3]:
1114 release = 'Vista'
1115 else:
1116 release = ''
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001117
1118 # In case we still don't know anything useful, we'll try to
1119 # help ourselves
1120 if system in ('win32','win16'):
1121 if not version:
1122 if system == 'win32':
1123 version = '32bit'
1124 else:
1125 version = '16bit'
1126 system = 'Windows'
1127
1128 elif system[:4] == 'java':
1129 release,vendor,vminfo,osinfo = java_ver()
1130 system = 'Java'
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001131 version = ', '.join(vminfo)
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001132 if not version:
1133 version = vendor
1134
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001135 # System specific extensions
1136 if system == 'OpenVMS':
1137 # OpenVMS seems to have release and version mixed up
1138 if not release or release == '0':
1139 release = version
1140 version = ''
1141 # Get processor information
1142 try:
1143 import vms_lib
1144 except ImportError:
1145 pass
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001146 else:
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001147 csid, cpu_number = vms_lib.getsyi('SYI$_CPU',0)
1148 if (cpu_number >= 128):
1149 processor = 'Alpha'
1150 else:
1151 processor = 'VAX'
1152 if not processor:
1153 # Get processor information from the uname system command
1154 processor = _syscmd_uname('-p','')
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001155
Amaury Forgeot d'Arc35c86582008-06-17 21:11:29 +00001156 #If any unknowns still exist, replace them with ''s, which are more portable
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001157 if system == 'unknown':
1158 system = ''
1159 if node == 'unknown':
1160 node = ''
1161 if release == 'unknown':
1162 release = ''
1163 if version == 'unknown':
1164 version = ''
1165 if machine == 'unknown':
1166 machine = ''
1167 if processor == 'unknown':
1168 processor = ''
Thomas Wouters1b7f8912007-09-19 03:06:30 +00001169
1170 # normalize name
1171 if system == 'Microsoft' and release == 'Windows':
1172 system = 'Windows'
1173 release = 'Vista'
1174
Larry Hastings68386bc2012-06-24 14:30:41 -07001175 _uname_cache = uname_result(system,node,release,version,machine,processor)
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001176 return _uname_cache
1177
1178### Direct interfaces to some of the uname() return values
1179
1180def system():
1181
1182 """ Returns the system/OS name, e.g. 'Linux', 'Windows' or 'Java'.
1183
1184 An empty string is returned if the value cannot be determined.
1185
1186 """
Larry Hastings68386bc2012-06-24 14:30:41 -07001187 return uname().system
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001188
1189def node():
1190
Brett Cannon8ab27df2003-08-05 03:52:04 +00001191 """ Returns the computer's network name (which may not be fully
1192 qualified)
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001193
1194 An empty string is returned if the value cannot be determined.
1195
1196 """
Larry Hastings68386bc2012-06-24 14:30:41 -07001197 return uname().node
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001198
1199def release():
1200
1201 """ Returns the system's release, e.g. '2.2.0' or 'NT'
1202
1203 An empty string is returned if the value cannot be determined.
1204
1205 """
Larry Hastings68386bc2012-06-24 14:30:41 -07001206 return uname().release
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001207
1208def version():
1209
1210 """ Returns the system's release version, e.g. '#3 on degas'
1211
1212 An empty string is returned if the value cannot be determined.
1213
1214 """
Larry Hastings68386bc2012-06-24 14:30:41 -07001215 return uname().version
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001216
1217def machine():
1218
1219 """ Returns the machine type, e.g. 'i386'
1220
1221 An empty string is returned if the value cannot be determined.
1222
1223 """
Larry Hastings68386bc2012-06-24 14:30:41 -07001224 return uname().machine
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001225
1226def processor():
1227
1228 """ Returns the (true) processor name, e.g. 'amdk6'
1229
1230 An empty string is returned if the value cannot be
1231 determined. Note that many platforms do not provide this
1232 information or simply return the same value as for machine(),
1233 e.g. NetBSD does this.
1234
1235 """
Larry Hastings68386bc2012-06-24 14:30:41 -07001236 return uname().processor
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001237
1238### Various APIs for extracting information from sys.version
1239
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001240_sys_version_parser = re.compile(
1241 r'([\w.+]+)\s*'
1242 '\(#?([^,]+),\s*([\w ]+),\s*([\w :]+)\)\s*'
Antoine Pitroufd036452008-08-19 17:56:33 +00001243 '\[([^\]]+)\]?', re.ASCII)
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001244
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001245_ironpython_sys_version_parser = re.compile(
1246 r'IronPython\s*'
1247 '([\d\.]+)'
1248 '(?: \(([\d\.]+)\))?'
Antoine Pitroufd036452008-08-19 17:56:33 +00001249 ' on (.NET [\d\.]+)', re.ASCII)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001250
Benjamin Petersone549ead2009-03-28 21:42:05 +00001251_pypy_sys_version_parser = re.compile(
1252 r'([\w.+]+)\s*'
1253 '\(#?([^,]+),\s*([\w ]+),\s*([\w :]+)\)\s*'
1254 '\[PyPy [^\]]+\]?')
1255
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001256_sys_version_cache = {}
1257
1258def _sys_version(sys_version=None):
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001259
1260 """ Returns a parsed version of Python's sys.version as tuple
Benjamin Peterson5f28b7b2009-03-26 21:49:58 +00001261 (name, version, branch, revision, buildno, builddate, compiler)
1262 referring to the Python implementation name, version, branch,
1263 revision, build number, build date/time as string and the compiler
1264 identification string.
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001265
1266 Note that unlike the Python sys.version, the returned value
1267 for the Python version will always include the patchlevel (it
1268 defaults to '.0').
1269
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001270 The function returns empty strings for tuple entries that
1271 cannot be determined.
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001272
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001273 sys_version may be given to parse an alternative version
1274 string, e.g. if the version was read from a different Python
1275 interpreter.
1276
1277 """
1278 # Get the Python version
1279 if sys_version is None:
1280 sys_version = sys.version
1281
1282 # Try the cache first
1283 result = _sys_version_cache.get(sys_version, None)
1284 if result is not None:
1285 return result
1286
1287 # Parse it
1288 if sys_version[:10] == 'IronPython':
1289 # IronPython
1290 name = 'IronPython'
1291 match = _ironpython_sys_version_parser.match(sys_version)
1292 if match is None:
1293 raise ValueError(
1294 'failed to parse IronPython sys.version: %s' %
1295 repr(sys_version))
1296 version, alt_version, compiler = match.groups()
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001297 buildno = ''
1298 builddate = ''
1299
1300 elif sys.platform[:4] == 'java':
1301 # Jython
1302 name = 'Jython'
Benjamin Petersone549ead2009-03-28 21:42:05 +00001303 match = _sys_version_parser.match(sys_version)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001304 if match is None:
1305 raise ValueError(
1306 'failed to parse Jython sys.version: %s' %
1307 repr(sys_version))
Benjamin Petersone549ead2009-03-28 21:42:05 +00001308 version, buildno, builddate, buildtime, _ = match.groups()
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001309 compiler = sys.platform
Benjamin Petersone549ead2009-03-28 21:42:05 +00001310
1311 elif "PyPy" in sys_version:
1312 # PyPy
1313 name = "PyPy"
1314 match = _pypy_sys_version_parser.match(sys_version)
1315 if match is None:
1316 raise ValueError("failed to parse PyPy sys.version: %s" %
1317 repr(sys_version))
1318 version, buildno, builddate, buildtime = match.groups()
1319 compiler = ""
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001320
1321 else:
1322 # CPython
1323 match = _sys_version_parser.match(sys_version)
1324 if match is None:
1325 raise ValueError(
1326 'failed to parse CPython sys.version: %s' %
1327 repr(sys_version))
1328 version, buildno, builddate, buildtime, compiler = \
1329 match.groups()
Benjamin Petersone549ead2009-03-28 21:42:05 +00001330 name = 'CPython'
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001331 builddate = builddate + ' ' + buildtime
1332
Georg Brandl82562422011-03-05 21:09:22 +01001333 if hasattr(sys, '_mercurial'):
1334 _, branch, revision = sys._mercurial
1335 elif hasattr(sys, 'subversion'):
Benjamin Petersone549ead2009-03-28 21:42:05 +00001336 # sys.subversion was added in Python 2.5
1337 _, branch, revision = sys.subversion
1338 else:
1339 branch = ''
1340 revision = ''
1341
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001342 # Add the patchlevel version if missing
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001343 l = version.split('.')
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001344 if len(l) == 2:
1345 l.append('0')
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001346 version = '.'.join(l)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001347
1348 # Build and cache the result
1349 result = (name, version, branch, revision, buildno, builddate, compiler)
1350 _sys_version_cache[sys_version] = result
1351 return result
1352
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001353def python_implementation():
1354
1355 """ Returns a string identifying the Python implementation.
1356
1357 Currently, the following implementations are identified:
Ezio Melottif16898b2011-05-04 18:37:50 +03001358 'CPython' (C implementation of Python),
1359 'IronPython' (.NET implementation of Python),
1360 'Jython' (Java implementation of Python),
1361 'PyPy' (Python implementation of Python).
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001362
1363 """
1364 return _sys_version()[0]
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001365
1366def python_version():
1367
1368 """ Returns the Python version as string 'major.minor.patchlevel'
1369
1370 Note that unlike the Python sys.version, the returned value
1371 will always include the patchlevel (it defaults to 0).
1372
1373 """
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001374 return _sys_version()[1]
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001375
1376def python_version_tuple():
1377
1378 """ Returns the Python version as tuple (major, minor, patchlevel)
1379 of strings.
1380
1381 Note that unlike the Python sys.version, the returned value
1382 will always include the patchlevel (it defaults to 0).
1383
1384 """
Neal Norwitz9d72bb42007-04-17 08:48:32 +00001385 return tuple(_sys_version()[1].split('.'))
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001386
1387def python_branch():
1388
1389 """ Returns a string identifying the Python implementation
1390 branch.
1391
1392 For CPython this is the Subversion branch from which the
1393 Python binary was built.
1394
1395 If not available, an empty string is returned.
1396
1397 """
Thomas Wouters9fe394c2007-02-05 01:24:16 +00001398
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001399 return _sys_version()[2]
1400
1401def python_revision():
1402
1403 """ Returns a string identifying the Python implementation
1404 revision.
1405
1406 For CPython this is the Subversion revision from which the
1407 Python binary was built.
1408
1409 If not available, an empty string is returned.
1410
1411 """
1412 return _sys_version()[3]
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001413
1414def python_build():
1415
1416 """ Returns a tuple (buildno, builddate) stating the Python
1417 build number and date as strings.
1418
1419 """
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001420 return _sys_version()[4:6]
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001421
1422def python_compiler():
1423
1424 """ Returns a string identifying the compiler used for compiling
1425 Python.
1426
1427 """
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001428 return _sys_version()[6]
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001429
1430### The Opus Magnum of platform strings :-)
1431
Marc-André Lemburg91e83e22004-03-25 18:35:12 +00001432_platform_cache = {}
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001433
1434def platform(aliased=0, terse=0):
1435
1436 """ Returns a single string identifying the underlying platform
1437 with as much useful information as possible (but no more :).
Tim Peters0eadaac2003-04-24 16:02:54 +00001438
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001439 The output is intended to be human readable rather than
1440 machine parseable. It may look different on different
1441 platforms and this is intended.
1442
1443 If "aliased" is true, the function will use aliases for
1444 various platforms that report system names which differ from
1445 their common names, e.g. SunOS will be reported as
1446 Solaris. The system_alias() function is used to implement
1447 this.
1448
1449 Setting terse to true causes the function to return only the
1450 absolute minimum information needed to identify the platform.
1451
1452 """
Marc-André Lemburg91e83e22004-03-25 18:35:12 +00001453 result = _platform_cache.get((aliased, terse), None)
1454 if result is not None:
1455 return result
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001456
1457 # Get uname information and then apply platform specific cosmetics
1458 # to it...
1459 system,node,release,version,machine,processor = uname()
1460 if machine == processor:
1461 processor = ''
1462 if aliased:
1463 system,release,version = system_alias(system,release,version)
1464
1465 if system == 'Windows':
1466 # MS platforms
1467 rel,vers,csd,ptype = win32_ver(version)
1468 if terse:
1469 platform = _platform(system,release)
1470 else:
1471 platform = _platform(system,release,version,csd)
1472
1473 elif system in ('Linux',):
1474 # Linux based systems
1475 distname,distversion,distid = dist('')
1476 if distname and not terse:
1477 platform = _platform(system,release,machine,processor,
1478 'with',
1479 distname,distversion,distid)
1480 else:
1481 # If the distribution name is unknown check for libc vs. glibc
1482 libcname,libcversion = libc_ver(sys.executable)
1483 platform = _platform(system,release,machine,processor,
1484 'with',
1485 libcname+libcversion)
1486 elif system == 'Java':
1487 # Java platforms
1488 r,v,vminfo,(os_name,os_version,os_arch) = java_ver()
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +00001489 if terse or not os_name:
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001490 platform = _platform(system,release,version)
1491 else:
1492 platform = _platform(system,release,version,
1493 'on',
1494 os_name,os_version,os_arch)
1495
1496 elif system == 'MacOS':
1497 # MacOS platforms
1498 if terse:
1499 platform = _platform(system,release)
1500 else:
1501 platform = _platform(system,release,machine)
1502
1503 else:
1504 # Generic handler
1505 if terse:
1506 platform = _platform(system,release)
1507 else:
1508 bits,linkage = architecture(sys.executable)
1509 platform = _platform(system,release,machine,processor,bits,linkage)
Tim Peters0eadaac2003-04-24 16:02:54 +00001510
Marc-André Lemburg91e83e22004-03-25 18:35:12 +00001511 _platform_cache[(aliased, terse)] = platform
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001512 return platform
1513
1514### Command line interface
1515
1516if __name__ == '__main__':
1517 # Default is to print the aliased verbose platform string
Tim Peters0eadaac2003-04-24 16:02:54 +00001518 terse = ('terse' in sys.argv or '--terse' in sys.argv)
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001519 aliased = (not 'nonaliased' in sys.argv and not '--nonaliased' in sys.argv)
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001520 print(platform(aliased,terse))
Marc-André Lemburg246d8472003-04-24 11:36:11 +00001521 sys.exit(0)