3 """ This module tries to retrieve as much platform-identifying data as
4 possible. It makes this information available via function APIs.
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.
11 # This module is maintained by Marc-Andre Lemburg <mal@egenix.com>.
12 # If you find problems, please submit bug reports/patches via the
13 # Python bug tracker (http://bugs.python.org) and assign them to "lemburg".
15 # Note: Please keep this module compatible to Python 1.5.2.
18 # * more support for WinCE
19 # * support for MS-DOS (PythonDX ?)
20 # * support for Amiga and other still unsupported platforms running Python
21 # * support for additional Linux distributions
23 # Many thanks to all those who helped adding platform-specific
24 # checks (in no particular order):
26 # Charles G Waldman, David Arnold, Gordon McMillan, Ben Darnell,
27 # Jeff Bauer, Cliff Crawford, Ivan Van Laningham, Josef
28 # Betancourt, Randall Hopper, Karl Putland, John Farrell, Greg
29 # Andruk, Just van Rossum, Thomas Heller, Mark R. Levinson, Mark
30 # Hammond, Bill Tutt, Hans Nowak, Uwe Zessin (OpenVMS support),
31 # Colin Kong, Trent Mick, Guido van Rossum, Anthony Baxter
35 # <see CVS and SVN checkin messages for history>
37 # 1.0.7 - added DEV_NULL
38 # 1.0.6 - added linux_distribution()
39 # 1.0.5 - fixed Java support to allow running the module on Jython
40 # 1.0.4 - added IronPython support
41 # 1.0.3 - added normalization of Windows system name
42 # 1.0.2 - added more Windows support
43 # 1.0.1 - reformatted to make doc.py happy
44 # 1.0.0 - reformatted a bit and checked into Python CVS
45 # 0.8.0 - added sys.version parser and various new access
46 # APIs (python_version(), python_compiler(), etc.)
47 # 0.7.2 - fixed architecture() to use sizeof(pointer) where available
48 # 0.7.1 - added support for Caldera OpenLinux
49 # 0.7.0 - some fixes for WinCE; untabified the source file
50 # 0.6.2 - support for OpenVMS - requires version 1.5.2-V006 or higher and
51 # vms_lib.getsyi() configured
52 # 0.6.1 - added code to prevent 'uname -p' on platforms which are
53 # known not to support it
54 # 0.6.0 - fixed win32_ver() to hopefully work on Win95,98,NT and Win2k;
55 # did some cleanup of the interfaces - some APIs have changed
56 # 0.5.5 - fixed another type in the MacOS code... should have
57 # used more coffee today ;-)
58 # 0.5.4 - fixed a few typos in the MacOS code
59 # 0.5.3 - added experimental MacOS support; added better popen()
60 # workarounds in _syscmd_ver() -- still not 100% elegant
62 # 0.5.2 - fixed uname() to return '' instead of 'unknown' in all
63 # return values (the system uname command tends to return
64 # 'unknown' instead of just leaving the field emtpy)
65 # 0.5.1 - included code for slackware dist; added exception handlers
66 # to cover up situations where platforms don't have os.popen
67 # (e.g. Mac) or fail on socket.gethostname(); fixed libc
69 # 0.5.0 - changed the API names referring to system commands to *syscmd*;
70 # added java_ver(); made syscmd_ver() a private
71 # API (was system_ver() in previous versions) -- use uname()
72 # instead; extended the win32_ver() to also return processor
74 # 0.4.0 - added win32_ver() and modified the platform() output for WinXX
75 # 0.3.4 - fixed a bug in _follow_symlinks()
76 # 0.3.3 - fixed popen() and "file" command invokation bugs
77 # 0.3.2 - added architecture() API and support for it in platform()
78 # 0.3.1 - fixed syscmd_ver() RE to support Windows NT
79 # 0.3.0 - added system alias support
80 # 0.2.3 - removed 'wince' again... oh well.
81 # 0.2.2 - added 'wince' to syscmd_ver() supported platforms
82 # 0.2.1 - added cache logic and changed the platform string format
83 # 0.2.0 - changed the API to use functions instead of module globals
84 # since some action take too long to be run on module import
85 # 0.1.0 - first release
87 # You can always get the latest version of this module at:
89 # http://www.egenix.com/files/python/platform.py
91 # If that URL should fail, try contacting the author.
94 Copyright (c) 1999-2000, Marc-Andre Lemburg; mailto:mal@lemburg.com
95 Copyright (c) 2000-2009, eGenix.com Software GmbH; mailto:info@egenix.com
97 Permission to use, copy, modify, and distribute this software and its
98 documentation for any purpose and without fee or royalty is hereby granted,
99 provided that the above copyright notice appear in all copies and that
100 both that copyright notice and this permission notice appear in
101 supporting documentation or portions thereof, including modifications,
104 EGENIX.COM SOFTWARE GMBH DISCLAIMS ALL WARRANTIES WITH REGARD TO
105 THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
106 FITNESS, IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL,
107 INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
108 FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
109 NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
110 WITH THE USE OR PERFORMANCE OF THIS SOFTWARE !
114 __version__
= '1.0.7'
116 import sys
,string
,os
,re
118 ### Globals & Constants
120 # Determine the platform's /dev/null device
122 DEV_NULL
= os
.devnull
123 except AttributeError:
124 # os.devnull was added in Python 2.4, so emulate it for earlier
126 if sys
.platform
in ('dos','win32','win16','os2'):
127 # Use the old CP/M NUL as device name
130 # Standard Unix uses /dev/null
131 DEV_NULL
= '/dev/null'
133 ### Platform specific APIs
135 _libc_search
= re
.compile(r
'(__libc_init)'
139 '(libc(_\w+)?\.so(?:\.(\d[0-9.]*))?)')
141 def libc_ver(executable
=sys
.executable
,lib
='',version
='',
145 """ Tries to determine the libc version that the file executable
146 (which defaults to the Python interpreter) is linked against.
148 Returns a tuple of strings (lib,version) which default to the
149 given parameters in case the lookup fails.
151 Note that the function has intimate knowledge of how different
152 libc versions add symbols to the executable and thus is probably
153 only useable for executables compiled using gcc.
155 The file is read and scanned in chunks of chunksize bytes.
158 if hasattr(os
.path
, 'realpath'):
159 # Python 2.2 introduced os.path.realpath(); it is used
160 # here to work around problems with Cygwin not being
161 # able to open symlinks for reading
162 executable
= os
.path
.realpath(executable
)
163 f
= open(executable
,'rb')
164 binary
= f
.read(chunksize
)
167 m
= _libc_search
.search(binary
,pos
)
169 binary
= f
.read(chunksize
)
174 libcinit
,glibc
,glibcversion
,so
,threads
,soversion
= m
.groups()
175 if libcinit
and not lib
:
180 version
= glibcversion
181 elif glibcversion
> version
:
182 version
= glibcversion
186 if soversion
> version
:
188 if threads
and version
[-len(threads
):] != threads
:
189 version
= version
+ threads
194 def _dist_try_harder(distname
,version
,id):
196 """ Tries some special tricks to get the distribution
197 information in case the default method fails.
199 Currently supports older SuSE Linux, Caldera OpenLinux and
200 Slackware Linux distributions.
203 if os
.path
.exists('/var/adm/inst-log/info'):
204 # SuSE Linux stores distribution information in that file
205 info
= open('/var/adm/inst-log/info').readlines()
208 tv
= string
.split(line
)
213 if tag
== 'MIN_DIST_VERSION':
214 version
= string
.strip(value
)
215 elif tag
== 'DIST_IDENT':
216 values
= string
.split(value
,'-')
218 return distname
,version
,id
220 if os
.path
.exists('/etc/.installed'):
221 # Caldera OpenLinux has some infos in that file (thanks to Colin Kong)
222 info
= open('/etc/.installed').readlines()
224 pkg
= string
.split(line
,'-')
225 if len(pkg
) >= 2 and pkg
[0] == 'OpenLinux':
226 # XXX does Caldera support non Intel platforms ? If yes,
227 # where can we find the needed id ?
228 return 'OpenLinux',pkg
[1],id
230 if os
.path
.isdir('/usr/lib/setup'):
231 # Check for slackware verson tag file (thanks to Greg Andruk)
232 verfiles
= os
.listdir('/usr/lib/setup')
233 for n
in range(len(verfiles
)-1, -1, -1):
234 if verfiles
[n
][:14] != 'slack-version-':
238 distname
= 'slackware'
239 version
= verfiles
[-1][14:]
240 return distname
,version
,id
242 return distname
,version
,id
244 _release_filename
= re
.compile(r
'(\w+)[-_](release|version)')
245 _lsb_release_version
= re
.compile(r
'(.+)'
248 '[^(]*(?:\((.+)\))?')
249 _release_version
= re
.compile(r
'([^0-9]+)'
252 '[^(]*(?:\((.+)\))?')
254 # See also http://www.novell.com/coolsolutions/feature/11251.html
255 # and http://linuxmafia.com/faq/Admin/release-files.html
256 # and http://data.linux-ntfs.org/rpm/whichrpm
257 # and http://www.die.net/doc/linux/man/man1/lsb_release.1.html
260 'SuSE', 'debian', 'fedora', 'redhat', 'centos',
261 'mandrake', 'mandriva', 'rocks', 'slackware', 'yellowdog', 'gentoo',
262 'UnitedLinux', 'turbolinux')
264 def _parse_release_file(firstline
):
266 # Parse the first line
267 m
= _lsb_release_version
.match(firstline
)
269 # LSB format: "distro release x.x (codename)"
270 return tuple(m
.groups())
272 # Pre-LSB format: "distro x.x (codename)"
273 m
= _release_version
.match(firstline
)
275 return tuple(m
.groups())
277 # Unkown format... take the first two words
278 l
= string
.split(string
.strip(firstline
))
285 return '', version
, id
287 def linux_distribution(distname
='', version
='', id='',
289 supported_dists
=_supported_dists
,
290 full_distribution_name
=1):
292 """ Tries to determine the name of the Linux OS distribution name.
294 The function first looks for a distribution release file in
295 /etc and then reverts to _dist_try_harder() in case no
296 suitable files are found.
298 supported_dists may be given to define the set of Linux
299 distributions to look for. It defaults to a list of currently
300 supported Linux distributions identified by their release file
303 If full_distribution_name is true (default), the full
304 distribution read from the OS is returned. Otherwise the short
305 name taken from supported_dists is used.
307 Returns a tuple (distname,version,id) which default to the
308 args given as parameters.
312 etc
= os
.listdir('/etc')
314 # Probably not a Unix system
315 return distname
,version
,id
318 m
= _release_filename
.match(file)
320 _distname
,dummy
= m
.groups()
321 if _distname
in supported_dists
:
325 return _dist_try_harder(distname
,version
,id)
327 # Read the first line
328 f
= open('/etc/'+file, 'r')
329 firstline
= f
.readline()
331 _distname
, _version
, _id
= _parse_release_file(firstline
)
333 if _distname
and full_distribution_name
:
339 return distname
, version
, id
341 # To maintain backwards compatibility:
343 def dist(distname
='',version
='',id='',
345 supported_dists
=_supported_dists
):
347 """ Tries to determine the name of the Linux OS distribution name.
349 The function first looks for a distribution release file in
350 /etc and then reverts to _dist_try_harder() in case no
351 suitable files are found.
353 Returns a tuple (distname,version,id) which default to the
354 args given as parameters.
357 return linux_distribution(distname
, version
, id,
358 supported_dists
=supported_dists
,
359 full_distribution_name
=0)
363 """ Fairly portable (alternative) popen implementation.
365 This is mostly needed in case os.popen() is not available, or
366 doesn't work as advertised, e.g. in Win9X GUI programs like
369 Writing to the pipe is currently not supported.
377 def __init__(self
,cmd
,mode
='r',bufsize
=None):
380 raise ValueError,'popen()-emulation only supports read mode'
382 self
.tmpfile
= tmpfile
= tempfile
.mktemp()
383 os
.system(cmd
+ ' > %s' % tmpfile
)
384 self
.pipe
= open(tmpfile
,'rb')
385 self
.bufsize
= bufsize
390 return self
.pipe
.read()
394 if self
.bufsize
is not None:
395 return self
.pipe
.readlines()
399 remove
=os
.unlink
,error
=os
.error
):
402 rc
= self
.pipe
.close()
415 def popen(cmd
, mode
='r', bufsize
=None):
417 """ Portable popen() interface.
419 # Find a working popen implementation preferring win32pipe.popen
420 # over os.popen over _popen
422 if os
.environ
.get('OS','') == 'Windows_NT':
423 # On NT win32pipe should work; on Win9x it hangs due to bugs
424 # in the MS C lib (see MS KnowledgeBase article Q150956)
430 popen
= win32pipe
.popen
432 if hasattr(os
,'popen'):
434 # Check whether it works... it doesn't in GUI programs
435 # on Windows platforms
436 if sys
.platform
== 'win32': # XXX Others too ?
444 return popen(cmd
,mode
)
446 return popen(cmd
,mode
,bufsize
)
448 def _norm_version(version
, build
=''):
450 """ Normalize the version and build strings and return a single
451 version string using the format major.minor.build (or patchlevel).
453 l
= string
.split(version
,'.')
461 strings
= map(str,ints
)
462 version
= string
.join(strings
[:3],'.')
465 _ver_output
= re
.compile(r
'(?:([\w ]+) ([\w.]+) '
469 # Examples of VER command output:
471 # Windows 2000: Microsoft Windows 2000 [Version 5.00.2195]
472 # Windows XP: Microsoft Windows XP [Version 5.1.2600]
473 # Windows Vista: Microsoft Windows [Version 6.0.6002]
475 # Note that the "Version" string gets localized on different
478 def _syscmd_ver(system
='', release
='', version
='',
480 supported_platforms
=('win32','win16','dos','os2')):
482 """ Tries to figure out the OS version used and returns
483 a tuple (system,release,version).
485 It uses the "ver" shell command for this which is known
486 to exists on Windows, DOS and OS/2. XXX Others too ?
488 In case this fails, the given parameters are used as
492 if sys
.platform
not in supported_platforms
:
493 return system
,release
,version
495 # Try some common cmd strings
496 for cmd
in ('ver','command /c ver','cmd /c ver'):
501 raise os
.error
,'command failed'
502 # XXX How can I supress shell errors from being written
505 #print 'Command %s failed: %s' % (cmd,why)
508 #print 'Command %s failed: %s' % (cmd,why)
513 return system
,release
,version
516 info
= string
.strip(info
)
517 m
= _ver_output
.match(info
)
519 system
,release
,version
= m
.groups()
520 # Strip trailing dots from version and release
521 if release
[-1] == '.':
522 release
= release
[:-1]
523 if version
[-1] == '.':
524 version
= version
[:-1]
525 # Normalize the version and build strings (eliminating additional
527 version
= _norm_version(version
)
528 return system
,release
,version
530 def _win32_getvalue(key
,name
,default
=''):
532 """ Read a value for name from the registry key.
534 In case this fails, default is returned.
538 # Use win32api if available
539 from win32api
import RegQueryValueEx
541 # On Python 2.0 and later, emulate using _winreg
543 RegQueryValueEx
= _winreg
.QueryValueEx
545 return RegQueryValueEx(key
,name
)
549 def win32_ver(release
='',version
='',csd
='',ptype
=''):
551 """ Get additional version information from the Windows Registry
552 and return a tuple (version,csd,ptype) referring to version
553 number, CSD level and OS type (multi/single
556 As a hint: ptype returns 'Uniprocessor Free' on single
557 processor NT machines and 'Multiprocessor Free' on multi
558 processor machines. The 'Free' refers to the OS version being
559 free of debugging code. It could also state 'Checked' which
560 means the OS version uses debugging code, i.e. code that
561 checks arguments, ranges, etc. (Thomas Heller).
563 Note: this function works best with Mark Hammond's win32
564 package installed, but also on Python 2.3 and later. It
565 obviously only runs on Win32 compatible platforms.
568 # XXX Is there any way to find out the processor type on WinXX ?
569 # XXX Is win32 available on Windows CE ?
571 # Adapted from code posted by Karl Putland to comp.lang.python.
573 # The mappings between reg. values and release names can be found
574 # here: http://msdn.microsoft.com/library/en-us/sysinfo/base/osversioninfo_str.asp
576 # Import the needed APIs
579 from win32api
import RegQueryValueEx
, RegOpenKeyEx
, \
580 RegCloseKey
, GetVersionEx
581 from win32con
import HKEY_LOCAL_MACHINE
, VER_PLATFORM_WIN32_NT
, \
582 VER_PLATFORM_WIN32_WINDOWS
, VER_NT_WORKSTATION
584 # Emulate the win32api module using Python APIs
586 sys
.getwindowsversion
587 except AttributeError:
588 # No emulation possible, so return the defaults...
589 return release
,version
,csd
,ptype
591 # Emulation using _winreg (added in Python 2.0) and
592 # sys.getwindowsversion() (added in Python 2.3)
594 GetVersionEx
= sys
.getwindowsversion
595 RegQueryValueEx
= _winreg
.QueryValueEx
596 RegOpenKeyEx
= _winreg
.OpenKeyEx
597 RegCloseKey
= _winreg
.CloseKey
598 HKEY_LOCAL_MACHINE
= _winreg
.HKEY_LOCAL_MACHINE
599 VER_PLATFORM_WIN32_WINDOWS
= 1
600 VER_PLATFORM_WIN32_NT
= 2
601 VER_NT_WORKSTATION
= 1
603 # Find out the registry key and some general version infos
604 maj
,min,buildno
,plat
,csd
= GetVersionEx()
605 version
= '%i.%i.%i' % (maj
,min,buildno
& 0xFFFF)
606 if csd
[:13] == 'Service Pack ':
607 csd
= 'SP' + csd
[13:]
609 if plat
== VER_PLATFORM_WIN32_WINDOWS
:
610 regkey
= 'SOFTWARE\\Microsoft\\Windows\\CurrentVersion'
611 # Try to guess the release name
624 elif plat
== VER_PLATFORM_WIN32_NT
:
625 regkey
= 'SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion'
634 release
= '2003Server'
639 # Per http://msdn2.microsoft.com/en-us/library/ms724429.aspx
641 productType
= GetVersionEx(1)[8]
643 # sys.getwindowsversion() doesn't take any arguments, so
644 # we cannot detect 2008 Server that way.
645 # XXX Add some other means of detecting 2008 Server ?!
648 if productType
== VER_NT_WORKSTATION
:
651 release
= '2008Server'
653 # # Windows 7 release candidate uses version 6.1.7100
656 release
= 'post2008Server'
660 # E.g. Win3.1 with win32s
661 release
= '%i.%i' % (maj
,min)
662 return release
,version
,csd
,ptype
664 # Open the registry key
666 keyCurVer
= RegOpenKeyEx(HKEY_LOCAL_MACHINE
, regkey
)
667 # Get a value to make sure the key exists...
668 RegQueryValueEx(keyCurVer
, 'SystemRoot')
670 return release
,version
,csd
,ptype
673 #subversion = _win32_getvalue(keyCurVer,
674 # 'SubVersionNumber',
677 # release = release + subversion # 95a, 95b, etc.
678 build
= _win32_getvalue(keyCurVer
,
679 'CurrentBuildNumber',
681 ptype
= _win32_getvalue(keyCurVer
,
686 version
= _norm_version(version
,build
)
689 RegCloseKey(keyCurVer
)
690 return release
,version
,csd
,ptype
692 def _mac_ver_lookup(selectors
,default
=None):
694 from gestalt
import gestalt
698 for selector
in selectors
:
700 append(gestalt(selector
))
701 except (RuntimeError, MacOS
.Error
):
709 def mac_ver(release
='',versioninfo
=('','',''),machine
=''):
711 """ Get MacOS version information and return it as tuple (release,
712 versioninfo, machine) with versioninfo being a tuple (version,
713 dev_stage, non_release_version).
715 Entries which cannot be determined are set to the paramter values
716 which default to ''. All tuple entries are strings.
718 Thanks to Mark R. Levinson for mailing documentation links and
719 code examples for this function. Documentation for the
720 gestalt() API is available online at:
722 http://www.rgaros.nl/gestalt/
725 # Check whether the version info module is available
730 return release
,versioninfo
,machine
732 sysv
,sysu
,sysa
= _mac_ver_lookup(('sysv','sysu','sysa'))
735 major
= (sysv
& 0xFF00) >> 8
736 minor
= (sysv
& 0x00F0) >> 4
737 patch
= (sysv
& 0x000F)
739 if (major
, minor
) >= (10, 4):
740 # the 'sysv' gestald cannot return patchlevels
741 # higher than 9. Apple introduced 3 new
742 # gestalt codes in 10.4 to deal with this
743 # issue (needed because patch levels can
744 # run higher than 9, such as 10.4.11)
745 major
,minor
,patch
= _mac_ver_lookup(('sys1','sys2','sys3'))
746 release
= '%i.%i.%i' %(major
, minor
, patch
)
748 release
= '%s.%i.%i' % (_bcd2str(major
),minor
,patch
)
751 # NOTE: this block is left as documentation of the
752 # intention of this function, the 'sysu' gestalt is no
753 # longer available and there are no alternatives.
754 major
= int((sysu
& 0xFF000000L
) >> 24)
755 minor
= (sysu
& 0x00F00000) >> 20
756 bugfix
= (sysu
& 0x000F0000) >> 16
757 stage
= (sysu
& 0x0000FF00) >> 8
758 nonrel
= (sysu
& 0x000000FF)
759 version
= '%s.%i.%i' % (_bcd2str(major
),minor
,bugfix
)
760 nonrel
= _bcd2str(nonrel
)
761 stage
= {0x20:'development',
764 0x80:'final'}.get(stage
,'')
765 versioninfo
= (version
,stage
,nonrel
)
769 machine
= {0x1: '68k',
771 0xa: 'i386'}.get(sysa
,'')
772 return release
,versioninfo
,machine
774 def _java_getprop(name
,default
):
776 from java
.lang
import System
778 value
= System
.getProperty(name
)
782 except AttributeError:
785 def java_ver(release
='',vendor
='',vminfo
=('','',''),osinfo
=('','','')):
787 """ Version interface for Jython.
789 Returns a tuple (release,vendor,vminfo,osinfo) with vminfo being
790 a tuple (vm_name,vm_release,vm_vendor) and osinfo being a
791 tuple (os_name,os_version,os_arch).
793 Values which cannot be determined are set to the defaults
794 given as parameters (which all default to '').
797 # Import the needed APIs
801 return release
,vendor
,vminfo
,osinfo
803 vendor
= _java_getprop('java.vendor', vendor
)
804 release
= _java_getprop('java.version', release
)
805 vm_name
, vm_release
, vm_vendor
= vminfo
806 vm_name
= _java_getprop('java.vm.name', vm_name
)
807 vm_vendor
= _java_getprop('java.vm.vendor', vm_vendor
)
808 vm_release
= _java_getprop('java.vm.version', vm_release
)
809 vminfo
= vm_name
, vm_release
, vm_vendor
810 os_name
, os_version
, os_arch
= osinfo
811 os_arch
= _java_getprop('java.os.arch', os_arch
)
812 os_name
= _java_getprop('java.os.name', os_name
)
813 os_version
= _java_getprop('java.os.version', os_version
)
814 osinfo
= os_name
, os_version
, os_arch
816 return release
, vendor
, vminfo
, osinfo
818 ### System name aliasing
820 def system_alias(system
,release
,version
):
822 """ Returns (system,release,version) aliased to common
823 marketing names used for some systems.
825 It also does some reordering of the information in some cases
826 where it would otherwise cause confusion.
829 if system
== 'Rhapsody':
830 # Apple's BSD derivative
831 # XXX How can we determine the marketing release number ?
832 return 'MacOS X Server',system
+release
,version
834 elif system
== 'SunOS':
837 # These releases use the old name SunOS
838 return system
,release
,version
839 # Modify release (marketing release = SunOS release - 3)
840 l
= string
.split(release
,'.')
849 release
= string
.join(l
,'.')
853 # XXX Whatever the new SunOS marketing name is...
856 elif system
== 'IRIX64':
857 # IRIX reports IRIX64 on platforms with 64-bit support; yet it
858 # is really a version and not a different platform, since 32-bit
859 # apps are also supported..
862 version
= version
+ ' (64bit)'
866 elif system
in ('win32','win16'):
867 # In case one of the other tricks
870 return system
,release
,version
872 ### Various internal helpers
874 def _platform(*args
):
876 """ Helper to format the platform string in a filename
877 compatible format e.g. "system-version-machine".
879 # Format the platform string
880 platform
= string
.join(
885 # Cleanup some possible filename obstacles...
886 replace
= string
.replace
887 platform
= replace(platform
,' ','_')
888 platform
= replace(platform
,'/','-')
889 platform
= replace(platform
,'\\','-')
890 platform
= replace(platform
,':','-')
891 platform
= replace(platform
,';','-')
892 platform
= replace(platform
,'"','-')
893 platform
= replace(platform
,'(','-')
894 platform
= replace(platform
,')','-')
896 # No need to report 'unknown' information...
897 platform
= replace(platform
,'unknown','')
899 # Fold '--'s and remove trailing '-'
901 cleaned
= replace(platform
,'--','-')
902 if cleaned
== platform
:
905 while platform
[-1] == '-':
906 platform
= platform
[:-1]
910 def _node(default
=''):
912 """ Helper to determine the node name of this machine.
920 return socket
.gethostname()
922 # Still not working...
925 # os.path.abspath is new in Python 1.5.2:
926 if not hasattr(os
.path
,'abspath'):
930 isabs
=os
.path
.isabs
,join
=os
.path
.join
,getcwd
=os
.getcwd
,
931 normpath
=os
.path
.normpath
):
934 path
= join(getcwd(), path
)
935 return normpath(path
)
939 _abspath
= os
.path
.abspath
941 def _follow_symlinks(filepath
):
943 """ In case filepath is a symlink, follow it until a
944 real file is reached.
946 filepath
= _abspath(filepath
)
947 while os
.path
.islink(filepath
):
948 filepath
= os
.path
.normpath(
949 os
.path
.join(os
.path
.dirname(filepath
),os
.readlink(filepath
)))
952 def _syscmd_uname(option
,default
=''):
954 """ Interface to the system's uname command.
956 if sys
.platform
in ('dos','win32','win16','os2'):
960 f
= os
.popen('uname %s 2> %s' % (option
, DEV_NULL
))
961 except (AttributeError,os
.error
):
963 output
= string
.strip(f
.read())
970 def _syscmd_file(target
,default
=''):
972 """ Interface to the system's file command.
974 The function uses the -b option of the file command to have it
975 ommit the filename in its output and if possible the -L option
976 to have the command follow symlinks. It returns default in
977 case the command should fail.
980 if sys
.platform
in ('dos','win32','win16','os2'):
983 target
= _follow_symlinks(target
)
985 f
= os
.popen('file "%s" 2> %s' % (target
, DEV_NULL
))
986 except (AttributeError,os
.error
):
988 output
= string
.strip(f
.read())
995 ### Information about the used architecture
997 # Default values for architecture; non-empty strings override the
998 # defaults given as parameters
999 _default_architecture
= {
1000 'win32': ('','WindowsPE'),
1001 'win16': ('','Windows'),
1002 'dos': ('','MSDOS'),
1005 _architecture_split
= re
.compile(r
'[\s,]').split
1007 def architecture(executable
=sys
.executable
,bits
='',linkage
=''):
1009 """ Queries the given executable (defaults to the Python interpreter
1010 binary) for various architecture information.
1012 Returns a tuple (bits,linkage) which contains information about
1013 the bit architecture and the linkage format used for the
1014 executable. Both values are returned as strings.
1016 Values that cannot be determined are returned as given by the
1017 parameter presets. If bits is given as '', the sizeof(pointer)
1018 (or sizeof(long) on Python version < 1.5.2) is used as
1019 indicator for the supported pointer size.
1021 The function relies on the system's "file" command to do the
1022 actual work. This is available on most if not all Unix
1023 platforms. On some non-Unix platforms where the "file" command
1024 does not exist and the executable is set to the Python interpreter
1025 binary defaults from _default_architecture are used.
1028 # Use the sizeof(pointer) as default number of bits if nothing
1029 # else is given as default.
1033 size
= struct
.calcsize('P')
1034 except struct
.error
:
1035 # Older installations can only query longs
1036 size
= struct
.calcsize('l')
1037 bits
= str(size
*8) + 'bit'
1039 # Get data from the 'file' system command
1041 output
= _syscmd_file(executable
, '')
1046 executable
== sys
.executable
:
1047 # "file" command did not return anything; we'll try to provide
1048 # some sensible defaults then...
1049 if _default_architecture
.has_key(sys
.platform
):
1050 b
,l
= _default_architecture
[sys
.platform
]
1057 # Split the output into a list of strings omitting the filename
1058 fileout
= _architecture_split(output
)[1:]
1060 if 'executable' not in fileout
:
1061 # Format not supported
1065 if '32-bit' in fileout
:
1067 elif 'N32' in fileout
:
1070 elif '64-bit' in fileout
:
1074 if 'ELF' in fileout
:
1076 elif 'PE' in fileout
:
1077 # E.g. Windows uses this format
1078 if 'Windows' in fileout
:
1079 linkage
= 'WindowsPE'
1082 elif 'COFF' in fileout
:
1084 elif 'MS-DOS' in fileout
:
1087 # XXX the A.OUT format also falls under this class...
1092 ### Portable uname() interface
1098 """ Fairly portable uname interface. Returns a tuple
1099 of strings (system,node,release,version,machine,processor)
1100 identifying the underlying platform.
1102 Note that unlike the os.uname function this also returns
1103 possible processor information as an additional tuple entry.
1105 Entries which cannot be determined are set to ''.
1111 if _uname_cache
is not None:
1116 # Get some infos from the builtin os.uname API...
1118 system
,node
,release
,version
,machine
= os
.uname()
1119 except AttributeError:
1122 if no_os_uname
or not filter(None, (system
, node
, release
, version
, machine
)):
1123 # Hmm, no there is either no uname or uname has returned
1124 #'unknowns'... we'll have to poke around the system then.
1126 system
= sys
.platform
1134 # Try win32_ver() on win32 platforms
1135 if system
== 'win32':
1136 release
,version
,csd
,ptype
= win32_ver()
1137 if release
and version
:
1139 # Try to use the PROCESSOR_* environment variables
1140 # available on Win XP and later; see
1141 # http://support.microsoft.com/kb/888731 and
1142 # http://www.geocities.com/rick_lively/MANUALS/ENV/MSWIN/PROCESSI.HTM
1144 machine
= os
.environ
.get('PROCESSOR_ARCHITECTURE', '')
1146 processor
= os
.environ
.get('PROCESSOR_IDENTIFIER', machine
)
1148 # Try the 'ver' system command available on some
1151 system
,release
,version
= _syscmd_ver(system
)
1152 # Normalize system to what win32_ver() normally returns
1153 # (_syscmd_ver() tends to return the vendor name as well)
1154 if system
== 'Microsoft Windows':
1156 elif system
== 'Microsoft' and release
== 'Windows':
1157 # Under Windows Vista and Windows Server 2008,
1158 # Microsoft changed the output of the ver command. The
1159 # release is no longer printed. This causes the
1160 # system and release to be misidentified.
1162 if '6.0' == version
[:3]:
1167 # In case we still don't know anything useful, we'll try to
1169 if system
in ('win32','win16'):
1171 if system
== 'win32':
1177 elif system
[:4] == 'java':
1178 release
,vendor
,vminfo
,osinfo
= java_ver()
1180 version
= string
.join(vminfo
,', ')
1184 elif os
.name
== 'mac':
1185 release
,(version
,stage
,nonrel
),machine
= mac_ver()
1188 # System specific extensions
1189 if system
== 'OpenVMS':
1190 # OpenVMS seems to have release and version mixed up
1191 if not release
or release
== '0':
1194 # Get processor information
1200 csid
, cpu_number
= vms_lib
.getsyi('SYI$_CPU',0)
1201 if (cpu_number
>= 128):
1206 # Get processor information from the uname system command
1207 processor
= _syscmd_uname('-p','')
1209 #If any unknowns still exist, replace them with ''s, which are more portable
1210 if system
== 'unknown':
1212 if node
== 'unknown':
1214 if release
== 'unknown':
1216 if version
== 'unknown':
1218 if machine
== 'unknown':
1220 if processor
== 'unknown':
1224 if system
== 'Microsoft' and release
== 'Windows':
1228 _uname_cache
= system
,node
,release
,version
,machine
,processor
1231 ### Direct interfaces to some of the uname() return values
1235 """ Returns the system/OS name, e.g. 'Linux', 'Windows' or 'Java'.
1237 An empty string is returned if the value cannot be determined.
1244 """ Returns the computer's network name (which may not be fully
1247 An empty string is returned if the value cannot be determined.
1254 """ Returns the system's release, e.g. '2.2.0' or 'NT'
1256 An empty string is returned if the value cannot be determined.
1263 """ Returns the system's release version, e.g. '#3 on degas'
1265 An empty string is returned if the value cannot be determined.
1272 """ Returns the machine type, e.g. 'i386'
1274 An empty string is returned if the value cannot be determined.
1281 """ Returns the (true) processor name, e.g. 'amdk6'
1283 An empty string is returned if the value cannot be
1284 determined. Note that many platforms do not provide this
1285 information or simply return the same value as for machine(),
1286 e.g. NetBSD does this.
1291 ### Various APIs for extracting information from sys.version
1293 _sys_version_parser
= re
.compile(
1295 '\(#?([^,]+),\s*([\w ]+),\s*([\w :]+)\)\s*'
1298 _ironpython_sys_version_parser
= re
.compile(
1301 '(?: \(([\d\.]+)\))?'
1302 ' on (.NET [\d\.]+)')
1304 _pypy_sys_version_parser
= re
.compile(
1306 '\(#?([^,]+),\s*([\w ]+),\s*([\w :]+)\)\s*'
1309 _sys_version_cache
= {}
1311 def _sys_version(sys_version
=None):
1313 """ Returns a parsed version of Python's sys.version as tuple
1314 (name, version, branch, revision, buildno, builddate, compiler)
1315 referring to the Python implementation name, version, branch,
1316 revision, build number, build date/time as string and the compiler
1317 identification string.
1319 Note that unlike the Python sys.version, the returned value
1320 for the Python version will always include the patchlevel (it
1323 The function returns empty strings for tuple entries that
1324 cannot be determined.
1326 sys_version may be given to parse an alternative version
1327 string, e.g. if the version was read from a different Python
1331 # Get the Python version
1332 if sys_version
is None:
1333 sys_version
= sys
.version
1335 # Try the cache first
1336 result
= _sys_version_cache
.get(sys_version
, None)
1337 if result
is not None:
1341 if sys_version
[:10] == 'IronPython':
1344 match
= _ironpython_sys_version_parser
.match(sys_version
)
1347 'failed to parse IronPython sys.version: %s' %
1349 version
, alt_version
, compiler
= match
.groups()
1353 elif sys
.platform
[:4] == 'java':
1356 match
= _sys_version_parser
.match(sys_version
)
1359 'failed to parse Jython sys.version: %s' %
1361 version
, buildno
, builddate
, buildtime
, _
= match
.groups()
1362 compiler
= sys
.platform
1364 elif "PyPy" in sys_version
:
1367 match
= _pypy_sys_version_parser
.match(sys_version
)
1369 raise ValueError("failed to parse PyPy sys.version: %s" %
1371 version
, buildno
, builddate
, buildtime
= match
.groups()
1376 match
= _sys_version_parser
.match(sys_version
)
1379 'failed to parse CPython sys.version: %s' %
1381 version
, buildno
, builddate
, buildtime
, compiler
= \
1384 builddate
= builddate
+ ' ' + buildtime
1386 if hasattr(sys
, 'subversion'):
1387 # sys.subversion was added in Python 2.5
1388 _
, branch
, revision
= sys
.subversion
1393 # Add the patchlevel version if missing
1394 l
= string
.split(version
, '.')
1397 version
= string
.join(l
, '.')
1399 # Build and cache the result
1400 result
= (name
, version
, branch
, revision
, buildno
, builddate
, compiler
)
1401 _sys_version_cache
[sys_version
] = result
1404 def python_implementation():
1406 """ Returns a string identifying the Python implementation.
1408 Currently, the following implementations are identified:
1409 'CPython' (C implementation of Python),
1410 'IronPython' (.NET implementation of Python),
1411 'Jython' (Java implementation of Python).
1414 return _sys_version()[0]
1416 def python_version():
1418 """ Returns the Python version as string 'major.minor.patchlevel'
1420 Note that unlike the Python sys.version, the returned value
1421 will always include the patchlevel (it defaults to 0).
1424 return _sys_version()[1]
1426 def python_version_tuple():
1428 """ Returns the Python version as tuple (major, minor, patchlevel)
1431 Note that unlike the Python sys.version, the returned value
1432 will always include the patchlevel (it defaults to 0).
1435 return tuple(string
.split(_sys_version()[1], '.'))
1437 def python_branch():
1439 """ Returns a string identifying the Python implementation
1442 For CPython this is the Subversion branch from which the
1443 Python binary was built.
1445 If not available, an empty string is returned.
1449 return _sys_version()[2]
1451 def python_revision():
1453 """ Returns a string identifying the Python implementation
1456 For CPython this is the Subversion revision from which the
1457 Python binary was built.
1459 If not available, an empty string is returned.
1462 return _sys_version()[3]
1466 """ Returns a tuple (buildno, builddate) stating the Python
1467 build number and date as strings.
1470 return _sys_version()[4:6]
1472 def python_compiler():
1474 """ Returns a string identifying the compiler used for compiling
1478 return _sys_version()[6]
1480 ### The Opus Magnum of platform strings :-)
1482 _platform_cache
= {}
1484 def platform(aliased
=0, terse
=0):
1486 """ Returns a single string identifying the underlying platform
1487 with as much useful information as possible (but no more :).
1489 The output is intended to be human readable rather than
1490 machine parseable. It may look different on different
1491 platforms and this is intended.
1493 If "aliased" is true, the function will use aliases for
1494 various platforms that report system names which differ from
1495 their common names, e.g. SunOS will be reported as
1496 Solaris. The system_alias() function is used to implement
1499 Setting terse to true causes the function to return only the
1500 absolute minimum information needed to identify the platform.
1503 result
= _platform_cache
.get((aliased
, terse
), None)
1504 if result
is not None:
1507 # Get uname information and then apply platform specific cosmetics
1509 system
,node
,release
,version
,machine
,processor
= uname()
1510 if machine
== processor
:
1513 system
,release
,version
= system_alias(system
,release
,version
)
1515 if system
== 'Windows':
1517 rel
,vers
,csd
,ptype
= win32_ver(version
)
1519 platform
= _platform(system
,release
)
1521 platform
= _platform(system
,release
,version
,csd
)
1523 elif system
in ('Linux',):
1524 # Linux based systems
1525 distname
,distversion
,distid
= dist('')
1526 if distname
and not terse
:
1527 platform
= _platform(system
,release
,machine
,processor
,
1529 distname
,distversion
,distid
)
1531 # If the distribution name is unknown check for libc vs. glibc
1532 libcname
,libcversion
= libc_ver(sys
.executable
)
1533 platform
= _platform(system
,release
,machine
,processor
,
1535 libcname
+libcversion
)
1536 elif system
== 'Java':
1538 r
,v
,vminfo
,(os_name
,os_version
,os_arch
) = java_ver()
1539 if terse
or not os_name
:
1540 platform
= _platform(system
,release
,version
)
1542 platform
= _platform(system
,release
,version
,
1544 os_name
,os_version
,os_arch
)
1546 elif system
== 'MacOS':
1549 platform
= _platform(system
,release
)
1551 platform
= _platform(system
,release
,machine
)
1556 platform
= _platform(system
,release
)
1558 bits
,linkage
= architecture(sys
.executable
)
1559 platform
= _platform(system
,release
,machine
,processor
,bits
,linkage
)
1561 _platform_cache
[(aliased
, terse
)] = platform
1564 ### Command line interface
1566 if __name__
== '__main__':
1567 # Default is to print the aliased verbose platform string
1568 terse
= ('terse' in sys
.argv
or '--terse' in sys
.argv
)
1569 aliased
= (not 'nonaliased' in sys
.argv
and not '--nonaliased' in sys
.argv
)
1570 print platform(aliased
,terse
)