1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399 |
- #!/usr/bin/env python
- # Copyright 2015,2016,2017 Nir Cohen
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- """
- The ``distro`` package (``distro`` stands for Linux Distribution) provides
- information about the Linux distribution it runs on, such as a reliable
- machine-readable distro ID, or version information.
- It is the recommended replacement for Python's original
- :py:func:`platform.linux_distribution` function, but it provides much more
- functionality. An alternative implementation became necessary because Python
- 3.5 deprecated this function, and Python 3.8 removed it altogether. Its
- predecessor function :py:func:`platform.dist` was already deprecated since
- Python 2.6 and removed in Python 3.8. Still, there are many cases in which
- access to OS distribution information is needed. See `Python issue 1322
- <https://bugs.python.org/issue1322>`_ for more information.
- """
- import argparse
- import json
- import logging
- import os
- import re
- import shlex
- import subprocess
- import sys
- import warnings
- from typing import (
- Any,
- Callable,
- Dict,
- Iterable,
- Optional,
- Sequence,
- TextIO,
- Tuple,
- Type,
- )
- try:
- from typing import TypedDict
- except ImportError:
- # Python 3.7
- TypedDict = dict
- __version__ = "1.8.0"
- class VersionDict(TypedDict):
- major: str
- minor: str
- build_number: str
- class InfoDict(TypedDict):
- id: str
- version: str
- version_parts: VersionDict
- like: str
- codename: str
- _UNIXCONFDIR = os.environ.get("UNIXCONFDIR", "/etc")
- _UNIXUSRLIBDIR = os.environ.get("UNIXUSRLIBDIR", "/usr/lib")
- _OS_RELEASE_BASENAME = "os-release"
- #: Translation table for normalizing the "ID" attribute defined in os-release
- #: files, for use by the :func:`distro.id` method.
- #:
- #: * Key: Value as defined in the os-release file, translated to lower case,
- #: with blanks translated to underscores.
- #:
- #: * Value: Normalized value.
- NORMALIZED_OS_ID = {
- "ol": "oracle", # Oracle Linux
- "opensuse-leap": "opensuse", # Newer versions of OpenSuSE report as opensuse-leap
- }
- #: Translation table for normalizing the "Distributor ID" attribute returned by
- #: the lsb_release command, for use by the :func:`distro.id` method.
- #:
- #: * Key: Value as returned by the lsb_release command, translated to lower
- #: case, with blanks translated to underscores.
- #:
- #: * Value: Normalized value.
- NORMALIZED_LSB_ID = {
- "enterpriseenterpriseas": "oracle", # Oracle Enterprise Linux 4
- "enterpriseenterpriseserver": "oracle", # Oracle Linux 5
- "redhatenterpriseworkstation": "rhel", # RHEL 6, 7 Workstation
- "redhatenterpriseserver": "rhel", # RHEL 6, 7 Server
- "redhatenterprisecomputenode": "rhel", # RHEL 6 ComputeNode
- }
- #: Translation table for normalizing the distro ID derived from the file name
- #: of distro release files, for use by the :func:`distro.id` method.
- #:
- #: * Key: Value as derived from the file name of a distro release file,
- #: translated to lower case, with blanks translated to underscores.
- #:
- #: * Value: Normalized value.
- NORMALIZED_DISTRO_ID = {
- "redhat": "rhel", # RHEL 6.x, 7.x
- }
- # Pattern for content of distro release file (reversed)
- _DISTRO_RELEASE_CONTENT_REVERSED_PATTERN = re.compile(
- r"(?:[^)]*\)(.*)\()? *(?:STL )?([\d.+\-a-z]*\d) *(?:esaeler *)?(.+)"
- )
- # Pattern for base file name of distro release file
- _DISTRO_RELEASE_BASENAME_PATTERN = re.compile(r"(\w+)[-_](release|version)$")
- # Base file names to be looked up for if _UNIXCONFDIR is not readable.
- _DISTRO_RELEASE_BASENAMES = [
- "SuSE-release",
- "arch-release",
- "base-release",
- "centos-release",
- "fedora-release",
- "gentoo-release",
- "mageia-release",
- "mandrake-release",
- "mandriva-release",
- "mandrivalinux-release",
- "manjaro-release",
- "oracle-release",
- "redhat-release",
- "rocky-release",
- "sl-release",
- "slackware-version",
- ]
- # Base file names to be ignored when searching for distro release file
- _DISTRO_RELEASE_IGNORE_BASENAMES = (
- "debian_version",
- "lsb-release",
- "oem-release",
- _OS_RELEASE_BASENAME,
- "system-release",
- "plesk-release",
- "iredmail-release",
- )
- def linux_distribution(full_distribution_name: bool = True) -> Tuple[str, str, str]:
- """
- .. deprecated:: 1.6.0
- :func:`distro.linux_distribution()` is deprecated. It should only be
- used as a compatibility shim with Python's
- :py:func:`platform.linux_distribution()`. Please use :func:`distro.id`,
- :func:`distro.version` and :func:`distro.name` instead.
- Return information about the current OS distribution as a tuple
- ``(id_name, version, codename)`` with items as follows:
- * ``id_name``: If *full_distribution_name* is false, the result of
- :func:`distro.id`. Otherwise, the result of :func:`distro.name`.
- * ``version``: The result of :func:`distro.version`.
- * ``codename``: The extra item (usually in parentheses) after the
- os-release version number, or the result of :func:`distro.codename`.
- The interface of this function is compatible with the original
- :py:func:`platform.linux_distribution` function, supporting a subset of
- its parameters.
- The data it returns may not exactly be the same, because it uses more data
- sources than the original function, and that may lead to different data if
- the OS distribution is not consistent across multiple data sources it
- provides (there are indeed such distributions ...).
- Another reason for differences is the fact that the :func:`distro.id`
- method normalizes the distro ID string to a reliable machine-readable value
- for a number of popular OS distributions.
- """
- warnings.warn(
- "distro.linux_distribution() is deprecated. It should only be used as a "
- "compatibility shim with Python's platform.linux_distribution(). Please use "
- "distro.id(), distro.version() and distro.name() instead.",
- DeprecationWarning,
- stacklevel=2,
- )
- return _distro.linux_distribution(full_distribution_name)
- def id() -> str:
- """
- Return the distro ID of the current distribution, as a
- machine-readable string.
- For a number of OS distributions, the returned distro ID value is
- *reliable*, in the sense that it is documented and that it does not change
- across releases of the distribution.
- This package maintains the following reliable distro ID values:
- ============== =========================================
- Distro ID Distribution
- ============== =========================================
- "ubuntu" Ubuntu
- "debian" Debian
- "rhel" RedHat Enterprise Linux
- "centos" CentOS
- "fedora" Fedora
- "sles" SUSE Linux Enterprise Server
- "opensuse" openSUSE
- "amzn" Amazon Linux
- "arch" Arch Linux
- "buildroot" Buildroot
- "cloudlinux" CloudLinux OS
- "exherbo" Exherbo Linux
- "gentoo" GenToo Linux
- "ibm_powerkvm" IBM PowerKVM
- "kvmibm" KVM for IBM z Systems
- "linuxmint" Linux Mint
- "mageia" Mageia
- "mandriva" Mandriva Linux
- "parallels" Parallels
- "pidora" Pidora
- "raspbian" Raspbian
- "oracle" Oracle Linux (and Oracle Enterprise Linux)
- "scientific" Scientific Linux
- "slackware" Slackware
- "xenserver" XenServer
- "openbsd" OpenBSD
- "netbsd" NetBSD
- "freebsd" FreeBSD
- "midnightbsd" MidnightBSD
- "rocky" Rocky Linux
- "aix" AIX
- "guix" Guix System
- ============== =========================================
- If you have a need to get distros for reliable IDs added into this set,
- or if you find that the :func:`distro.id` function returns a different
- distro ID for one of the listed distros, please create an issue in the
- `distro issue tracker`_.
- **Lookup hierarchy and transformations:**
- First, the ID is obtained from the following sources, in the specified
- order. The first available and non-empty value is used:
- * the value of the "ID" attribute of the os-release file,
- * the value of the "Distributor ID" attribute returned by the lsb_release
- command,
- * the first part of the file name of the distro release file,
- The so determined ID value then passes the following transformations,
- before it is returned by this method:
- * it is translated to lower case,
- * blanks (which should not be there anyway) are translated to underscores,
- * a normalization of the ID is performed, based upon
- `normalization tables`_. The purpose of this normalization is to ensure
- that the ID is as reliable as possible, even across incompatible changes
- in the OS distributions. A common reason for an incompatible change is
- the addition of an os-release file, or the addition of the lsb_release
- command, with ID values that differ from what was previously determined
- from the distro release file name.
- """
- return _distro.id()
- def name(pretty: bool = False) -> str:
- """
- Return the name of the current OS distribution, as a human-readable
- string.
- If *pretty* is false, the name is returned without version or codename.
- (e.g. "CentOS Linux")
- If *pretty* is true, the version and codename are appended.
- (e.g. "CentOS Linux 7.1.1503 (Core)")
- **Lookup hierarchy:**
- The name is obtained from the following sources, in the specified order.
- The first available and non-empty value is used:
- * If *pretty* is false:
- - the value of the "NAME" attribute of the os-release file,
- - the value of the "Distributor ID" attribute returned by the lsb_release
- command,
- - the value of the "<name>" field of the distro release file.
- * If *pretty* is true:
- - the value of the "PRETTY_NAME" attribute of the os-release file,
- - the value of the "Description" attribute returned by the lsb_release
- command,
- - the value of the "<name>" field of the distro release file, appended
- with the value of the pretty version ("<version_id>" and "<codename>"
- fields) of the distro release file, if available.
- """
- return _distro.name(pretty)
- def version(pretty: bool = False, best: bool = False) -> str:
- """
- Return the version of the current OS distribution, as a human-readable
- string.
- If *pretty* is false, the version is returned without codename (e.g.
- "7.0").
- If *pretty* is true, the codename in parenthesis is appended, if the
- codename is non-empty (e.g. "7.0 (Maipo)").
- Some distributions provide version numbers with different precisions in
- the different sources of distribution information. Examining the different
- sources in a fixed priority order does not always yield the most precise
- version (e.g. for Debian 8.2, or CentOS 7.1).
- Some other distributions may not provide this kind of information. In these
- cases, an empty string would be returned. This behavior can be observed
- with rolling releases distributions (e.g. Arch Linux).
- The *best* parameter can be used to control the approach for the returned
- version:
- If *best* is false, the first non-empty version number in priority order of
- the examined sources is returned.
- If *best* is true, the most precise version number out of all examined
- sources is returned.
- **Lookup hierarchy:**
- In all cases, the version number is obtained from the following sources.
- If *best* is false, this order represents the priority order:
- * the value of the "VERSION_ID" attribute of the os-release file,
- * the value of the "Release" attribute returned by the lsb_release
- command,
- * the version number parsed from the "<version_id>" field of the first line
- of the distro release file,
- * the version number parsed from the "PRETTY_NAME" attribute of the
- os-release file, if it follows the format of the distro release files.
- * the version number parsed from the "Description" attribute returned by
- the lsb_release command, if it follows the format of the distro release
- files.
- """
- return _distro.version(pretty, best)
- def version_parts(best: bool = False) -> Tuple[str, str, str]:
- """
- Return the version of the current OS distribution as a tuple
- ``(major, minor, build_number)`` with items as follows:
- * ``major``: The result of :func:`distro.major_version`.
- * ``minor``: The result of :func:`distro.minor_version`.
- * ``build_number``: The result of :func:`distro.build_number`.
- For a description of the *best* parameter, see the :func:`distro.version`
- method.
- """
- return _distro.version_parts(best)
- def major_version(best: bool = False) -> str:
- """
- Return the major version of the current OS distribution, as a string,
- if provided.
- Otherwise, the empty string is returned. The major version is the first
- part of the dot-separated version string.
- For a description of the *best* parameter, see the :func:`distro.version`
- method.
- """
- return _distro.major_version(best)
- def minor_version(best: bool = False) -> str:
- """
- Return the minor version of the current OS distribution, as a string,
- if provided.
- Otherwise, the empty string is returned. The minor version is the second
- part of the dot-separated version string.
- For a description of the *best* parameter, see the :func:`distro.version`
- method.
- """
- return _distro.minor_version(best)
- def build_number(best: bool = False) -> str:
- """
- Return the build number of the current OS distribution, as a string,
- if provided.
- Otherwise, the empty string is returned. The build number is the third part
- of the dot-separated version string.
- For a description of the *best* parameter, see the :func:`distro.version`
- method.
- """
- return _distro.build_number(best)
- def like() -> str:
- """
- Return a space-separated list of distro IDs of distributions that are
- closely related to the current OS distribution in regards to packaging
- and programming interfaces, for example distributions the current
- distribution is a derivative from.
- **Lookup hierarchy:**
- This information item is only provided by the os-release file.
- For details, see the description of the "ID_LIKE" attribute in the
- `os-release man page
- <http://www.freedesktop.org/software/systemd/man/os-release.html>`_.
- """
- return _distro.like()
- def codename() -> str:
- """
- Return the codename for the release of the current OS distribution,
- as a string.
- If the distribution does not have a codename, an empty string is returned.
- Note that the returned codename is not always really a codename. For
- example, openSUSE returns "x86_64". This function does not handle such
- cases in any special way and just returns the string it finds, if any.
- **Lookup hierarchy:**
- * the codename within the "VERSION" attribute of the os-release file, if
- provided,
- * the value of the "Codename" attribute returned by the lsb_release
- command,
- * the value of the "<codename>" field of the distro release file.
- """
- return _distro.codename()
- def info(pretty: bool = False, best: bool = False) -> InfoDict:
- """
- Return certain machine-readable information items about the current OS
- distribution in a dictionary, as shown in the following example:
- .. sourcecode:: python
- {
- 'id': 'rhel',
- 'version': '7.0',
- 'version_parts': {
- 'major': '7',
- 'minor': '0',
- 'build_number': ''
- },
- 'like': 'fedora',
- 'codename': 'Maipo'
- }
- The dictionary structure and keys are always the same, regardless of which
- information items are available in the underlying data sources. The values
- for the various keys are as follows:
- * ``id``: The result of :func:`distro.id`.
- * ``version``: The result of :func:`distro.version`.
- * ``version_parts -> major``: The result of :func:`distro.major_version`.
- * ``version_parts -> minor``: The result of :func:`distro.minor_version`.
- * ``version_parts -> build_number``: The result of
- :func:`distro.build_number`.
- * ``like``: The result of :func:`distro.like`.
- * ``codename``: The result of :func:`distro.codename`.
- For a description of the *pretty* and *best* parameters, see the
- :func:`distro.version` method.
- """
- return _distro.info(pretty, best)
- def os_release_info() -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information items
- from the os-release file data source of the current OS distribution.
- See `os-release file`_ for details about these information items.
- """
- return _distro.os_release_info()
- def lsb_release_info() -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information items
- from the lsb_release command data source of the current OS distribution.
- See `lsb_release command output`_ for details about these information
- items.
- """
- return _distro.lsb_release_info()
- def distro_release_info() -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information items
- from the distro release file data source of the current OS distribution.
- See `distro release file`_ for details about these information items.
- """
- return _distro.distro_release_info()
- def uname_info() -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information items
- from the distro release file data source of the current OS distribution.
- """
- return _distro.uname_info()
- def os_release_attr(attribute: str) -> str:
- """
- Return a single named information item from the os-release file data source
- of the current OS distribution.
- Parameters:
- * ``attribute`` (string): Key of the information item.
- Returns:
- * (string): Value of the information item, if the item exists.
- The empty string, if the item does not exist.
- See `os-release file`_ for details about these information items.
- """
- return _distro.os_release_attr(attribute)
- def lsb_release_attr(attribute: str) -> str:
- """
- Return a single named information item from the lsb_release command output
- data source of the current OS distribution.
- Parameters:
- * ``attribute`` (string): Key of the information item.
- Returns:
- * (string): Value of the information item, if the item exists.
- The empty string, if the item does not exist.
- See `lsb_release command output`_ for details about these information
- items.
- """
- return _distro.lsb_release_attr(attribute)
- def distro_release_attr(attribute: str) -> str:
- """
- Return a single named information item from the distro release file
- data source of the current OS distribution.
- Parameters:
- * ``attribute`` (string): Key of the information item.
- Returns:
- * (string): Value of the information item, if the item exists.
- The empty string, if the item does not exist.
- See `distro release file`_ for details about these information items.
- """
- return _distro.distro_release_attr(attribute)
- def uname_attr(attribute: str) -> str:
- """
- Return a single named information item from the distro release file
- data source of the current OS distribution.
- Parameters:
- * ``attribute`` (string): Key of the information item.
- Returns:
- * (string): Value of the information item, if the item exists.
- The empty string, if the item does not exist.
- """
- return _distro.uname_attr(attribute)
- try:
- from functools import cached_property
- except ImportError:
- # Python < 3.8
- class cached_property: # type: ignore
- """A version of @property which caches the value. On access, it calls the
- underlying function and sets the value in `__dict__` so future accesses
- will not re-call the property.
- """
- def __init__(self, f: Callable[[Any], Any]) -> None:
- self._fname = f.__name__
- self._f = f
- def __get__(self, obj: Any, owner: Type[Any]) -> Any:
- assert obj is not None, f"call {self._fname} on an instance"
- ret = obj.__dict__[self._fname] = self._f(obj)
- return ret
- class LinuxDistribution:
- """
- Provides information about a OS distribution.
- This package creates a private module-global instance of this class with
- default initialization arguments, that is used by the
- `consolidated accessor functions`_ and `single source accessor functions`_.
- By using default initialization arguments, that module-global instance
- returns data about the current OS distribution (i.e. the distro this
- package runs on).
- Normally, it is not necessary to create additional instances of this class.
- However, in situations where control is needed over the exact data sources
- that are used, instances of this class can be created with a specific
- distro release file, or a specific os-release file, or without invoking the
- lsb_release command.
- """
- def __init__(
- self,
- include_lsb: Optional[bool] = None,
- os_release_file: str = "",
- distro_release_file: str = "",
- include_uname: Optional[bool] = None,
- root_dir: Optional[str] = None,
- include_oslevel: Optional[bool] = None,
- ) -> None:
- """
- The initialization method of this class gathers information from the
- available data sources, and stores that in private instance attributes.
- Subsequent access to the information items uses these private instance
- attributes, so that the data sources are read only once.
- Parameters:
- * ``include_lsb`` (bool): Controls whether the
- `lsb_release command output`_ is included as a data source.
- If the lsb_release command is not available in the program execution
- path, the data source for the lsb_release command will be empty.
- * ``os_release_file`` (string): The path name of the
- `os-release file`_ that is to be used as a data source.
- An empty string (the default) will cause the default path name to
- be used (see `os-release file`_ for details).
- If the specified or defaulted os-release file does not exist, the
- data source for the os-release file will be empty.
- * ``distro_release_file`` (string): The path name of the
- `distro release file`_ that is to be used as a data source.
- An empty string (the default) will cause a default search algorithm
- to be used (see `distro release file`_ for details).
- If the specified distro release file does not exist, or if no default
- distro release file can be found, the data source for the distro
- release file will be empty.
- * ``include_uname`` (bool): Controls whether uname command output is
- included as a data source. If the uname command is not available in
- the program execution path the data source for the uname command will
- be empty.
- * ``root_dir`` (string): The absolute path to the root directory to use
- to find distro-related information files. Note that ``include_*``
- parameters must not be enabled in combination with ``root_dir``.
- * ``include_oslevel`` (bool): Controls whether (AIX) oslevel command
- output is included as a data source. If the oslevel command is not
- available in the program execution path the data source will be
- empty.
- Public instance attributes:
- * ``os_release_file`` (string): The path name of the
- `os-release file`_ that is actually used as a data source. The
- empty string if no distro release file is used as a data source.
- * ``distro_release_file`` (string): The path name of the
- `distro release file`_ that is actually used as a data source. The
- empty string if no distro release file is used as a data source.
- * ``include_lsb`` (bool): The result of the ``include_lsb`` parameter.
- This controls whether the lsb information will be loaded.
- * ``include_uname`` (bool): The result of the ``include_uname``
- parameter. This controls whether the uname information will
- be loaded.
- * ``include_oslevel`` (bool): The result of the ``include_oslevel``
- parameter. This controls whether (AIX) oslevel information will be
- loaded.
- * ``root_dir`` (string): The result of the ``root_dir`` parameter.
- The absolute path to the root directory to use to find distro-related
- information files.
- Raises:
- * :py:exc:`ValueError`: Initialization parameters combination is not
- supported.
- * :py:exc:`OSError`: Some I/O issue with an os-release file or distro
- release file.
- * :py:exc:`UnicodeError`: A data source has unexpected characters or
- uses an unexpected encoding.
- """
- self.root_dir = root_dir
- self.etc_dir = os.path.join(root_dir, "etc") if root_dir else _UNIXCONFDIR
- self.usr_lib_dir = (
- os.path.join(root_dir, "usr/lib") if root_dir else _UNIXUSRLIBDIR
- )
- if os_release_file:
- self.os_release_file = os_release_file
- else:
- etc_dir_os_release_file = os.path.join(self.etc_dir, _OS_RELEASE_BASENAME)
- usr_lib_os_release_file = os.path.join(
- self.usr_lib_dir, _OS_RELEASE_BASENAME
- )
- # NOTE: The idea is to respect order **and** have it set
- # at all times for API backwards compatibility.
- if os.path.isfile(etc_dir_os_release_file) or not os.path.isfile(
- usr_lib_os_release_file
- ):
- self.os_release_file = etc_dir_os_release_file
- else:
- self.os_release_file = usr_lib_os_release_file
- self.distro_release_file = distro_release_file or "" # updated later
- is_root_dir_defined = root_dir is not None
- if is_root_dir_defined and (include_lsb or include_uname or include_oslevel):
- raise ValueError(
- "Including subprocess data sources from specific root_dir is disallowed"
- " to prevent false information"
- )
- self.include_lsb = (
- include_lsb if include_lsb is not None else not is_root_dir_defined
- )
- self.include_uname = (
- include_uname if include_uname is not None else not is_root_dir_defined
- )
- self.include_oslevel = (
- include_oslevel if include_oslevel is not None else not is_root_dir_defined
- )
- def __repr__(self) -> str:
- """Return repr of all info"""
- return (
- "LinuxDistribution("
- "os_release_file={self.os_release_file!r}, "
- "distro_release_file={self.distro_release_file!r}, "
- "include_lsb={self.include_lsb!r}, "
- "include_uname={self.include_uname!r}, "
- "include_oslevel={self.include_oslevel!r}, "
- "root_dir={self.root_dir!r}, "
- "_os_release_info={self._os_release_info!r}, "
- "_lsb_release_info={self._lsb_release_info!r}, "
- "_distro_release_info={self._distro_release_info!r}, "
- "_uname_info={self._uname_info!r}, "
- "_oslevel_info={self._oslevel_info!r})".format(self=self)
- )
- def linux_distribution(
- self, full_distribution_name: bool = True
- ) -> Tuple[str, str, str]:
- """
- Return information about the OS distribution that is compatible
- with Python's :func:`platform.linux_distribution`, supporting a subset
- of its parameters.
- For details, see :func:`distro.linux_distribution`.
- """
- return (
- self.name() if full_distribution_name else self.id(),
- self.version(),
- self._os_release_info.get("release_codename") or self.codename(),
- )
- def id(self) -> str:
- """Return the distro ID of the OS distribution, as a string.
- For details, see :func:`distro.id`.
- """
- def normalize(distro_id: str, table: Dict[str, str]) -> str:
- distro_id = distro_id.lower().replace(" ", "_")
- return table.get(distro_id, distro_id)
- distro_id = self.os_release_attr("id")
- if distro_id:
- return normalize(distro_id, NORMALIZED_OS_ID)
- distro_id = self.lsb_release_attr("distributor_id")
- if distro_id:
- return normalize(distro_id, NORMALIZED_LSB_ID)
- distro_id = self.distro_release_attr("id")
- if distro_id:
- return normalize(distro_id, NORMALIZED_DISTRO_ID)
- distro_id = self.uname_attr("id")
- if distro_id:
- return normalize(distro_id, NORMALIZED_DISTRO_ID)
- return ""
- def name(self, pretty: bool = False) -> str:
- """
- Return the name of the OS distribution, as a string.
- For details, see :func:`distro.name`.
- """
- name = (
- self.os_release_attr("name")
- or self.lsb_release_attr("distributor_id")
- or self.distro_release_attr("name")
- or self.uname_attr("name")
- )
- if pretty:
- name = self.os_release_attr("pretty_name") or self.lsb_release_attr(
- "description"
- )
- if not name:
- name = self.distro_release_attr("name") or self.uname_attr("name")
- version = self.version(pretty=True)
- if version:
- name = f"{name} {version}"
- return name or ""
- def version(self, pretty: bool = False, best: bool = False) -> str:
- """
- Return the version of the OS distribution, as a string.
- For details, see :func:`distro.version`.
- """
- versions = [
- self.os_release_attr("version_id"),
- self.lsb_release_attr("release"),
- self.distro_release_attr("version_id"),
- self._parse_distro_release_content(self.os_release_attr("pretty_name")).get(
- "version_id", ""
- ),
- self._parse_distro_release_content(
- self.lsb_release_attr("description")
- ).get("version_id", ""),
- self.uname_attr("release"),
- ]
- if self.uname_attr("id").startswith("aix"):
- # On AIX platforms, prefer oslevel command output.
- versions.insert(0, self.oslevel_info())
- elif self.id() == "debian" or "debian" in self.like().split():
- # On Debian-like, add debian_version file content to candidates list.
- versions.append(self._debian_version)
- version = ""
- if best:
- # This algorithm uses the last version in priority order that has
- # the best precision. If the versions are not in conflict, that
- # does not matter; otherwise, using the last one instead of the
- # first one might be considered a surprise.
- for v in versions:
- if v.count(".") > version.count(".") or version == "":
- version = v
- else:
- for v in versions:
- if v != "":
- version = v
- break
- if pretty and version and self.codename():
- version = f"{version} ({self.codename()})"
- return version
- def version_parts(self, best: bool = False) -> Tuple[str, str, str]:
- """
- Return the version of the OS distribution, as a tuple of version
- numbers.
- For details, see :func:`distro.version_parts`.
- """
- version_str = self.version(best=best)
- if version_str:
- version_regex = re.compile(r"(\d+)\.?(\d+)?\.?(\d+)?")
- matches = version_regex.match(version_str)
- if matches:
- major, minor, build_number = matches.groups()
- return major, minor or "", build_number or ""
- return "", "", ""
- def major_version(self, best: bool = False) -> str:
- """
- Return the major version number of the current distribution.
- For details, see :func:`distro.major_version`.
- """
- return self.version_parts(best)[0]
- def minor_version(self, best: bool = False) -> str:
- """
- Return the minor version number of the current distribution.
- For details, see :func:`distro.minor_version`.
- """
- return self.version_parts(best)[1]
- def build_number(self, best: bool = False) -> str:
- """
- Return the build number of the current distribution.
- For details, see :func:`distro.build_number`.
- """
- return self.version_parts(best)[2]
- def like(self) -> str:
- """
- Return the IDs of distributions that are like the OS distribution.
- For details, see :func:`distro.like`.
- """
- return self.os_release_attr("id_like") or ""
- def codename(self) -> str:
- """
- Return the codename of the OS distribution.
- For details, see :func:`distro.codename`.
- """
- try:
- # Handle os_release specially since distros might purposefully set
- # this to empty string to have no codename
- return self._os_release_info["codename"]
- except KeyError:
- return (
- self.lsb_release_attr("codename")
- or self.distro_release_attr("codename")
- or ""
- )
- def info(self, pretty: bool = False, best: bool = False) -> InfoDict:
- """
- Return certain machine-readable information about the OS
- distribution.
- For details, see :func:`distro.info`.
- """
- return dict(
- id=self.id(),
- version=self.version(pretty, best),
- version_parts=dict(
- major=self.major_version(best),
- minor=self.minor_version(best),
- build_number=self.build_number(best),
- ),
- like=self.like(),
- codename=self.codename(),
- )
- def os_release_info(self) -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information
- items from the os-release file data source of the OS distribution.
- For details, see :func:`distro.os_release_info`.
- """
- return self._os_release_info
- def lsb_release_info(self) -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information
- items from the lsb_release command data source of the OS
- distribution.
- For details, see :func:`distro.lsb_release_info`.
- """
- return self._lsb_release_info
- def distro_release_info(self) -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information
- items from the distro release file data source of the OS
- distribution.
- For details, see :func:`distro.distro_release_info`.
- """
- return self._distro_release_info
- def uname_info(self) -> Dict[str, str]:
- """
- Return a dictionary containing key-value pairs for the information
- items from the uname command data source of the OS distribution.
- For details, see :func:`distro.uname_info`.
- """
- return self._uname_info
- def oslevel_info(self) -> str:
- """
- Return AIX' oslevel command output.
- """
- return self._oslevel_info
- def os_release_attr(self, attribute: str) -> str:
- """
- Return a single named information item from the os-release file data
- source of the OS distribution.
- For details, see :func:`distro.os_release_attr`.
- """
- return self._os_release_info.get(attribute, "")
- def lsb_release_attr(self, attribute: str) -> str:
- """
- Return a single named information item from the lsb_release command
- output data source of the OS distribution.
- For details, see :func:`distro.lsb_release_attr`.
- """
- return self._lsb_release_info.get(attribute, "")
- def distro_release_attr(self, attribute: str) -> str:
- """
- Return a single named information item from the distro release file
- data source of the OS distribution.
- For details, see :func:`distro.distro_release_attr`.
- """
- return self._distro_release_info.get(attribute, "")
- def uname_attr(self, attribute: str) -> str:
- """
- Return a single named information item from the uname command
- output data source of the OS distribution.
- For details, see :func:`distro.uname_attr`.
- """
- return self._uname_info.get(attribute, "")
- @cached_property
- def _os_release_info(self) -> Dict[str, str]:
- """
- Get the information items from the specified os-release file.
- Returns:
- A dictionary containing all information items.
- """
- if os.path.isfile(self.os_release_file):
- with open(self.os_release_file, encoding="utf-8") as release_file:
- return self._parse_os_release_content(release_file)
- return {}
- @staticmethod
- def _parse_os_release_content(lines: TextIO) -> Dict[str, str]:
- """
- Parse the lines of an os-release file.
- Parameters:
- * lines: Iterable through the lines in the os-release file.
- Each line must be a unicode string or a UTF-8 encoded byte
- string.
- Returns:
- A dictionary containing all information items.
- """
- props = {}
- lexer = shlex.shlex(lines, posix=True)
- lexer.whitespace_split = True
- tokens = list(lexer)
- for token in tokens:
- # At this point, all shell-like parsing has been done (i.e.
- # comments processed, quotes and backslash escape sequences
- # processed, multi-line values assembled, trailing newlines
- # stripped, etc.), so the tokens are now either:
- # * variable assignments: var=value
- # * commands or their arguments (not allowed in os-release)
- # Ignore any tokens that are not variable assignments
- if "=" in token:
- k, v = token.split("=", 1)
- props[k.lower()] = v
- if "version" in props:
- # extract release codename (if any) from version attribute
- match = re.search(r"\((\D+)\)|,\s*(\D+)", props["version"])
- if match:
- release_codename = match.group(1) or match.group(2)
- props["codename"] = props["release_codename"] = release_codename
- if "version_codename" in props:
- # os-release added a version_codename field. Use that in
- # preference to anything else Note that some distros purposefully
- # do not have code names. They should be setting
- # version_codename=""
- props["codename"] = props["version_codename"]
- elif "ubuntu_codename" in props:
- # Same as above but a non-standard field name used on older Ubuntus
- props["codename"] = props["ubuntu_codename"]
- return props
- @cached_property
- def _lsb_release_info(self) -> Dict[str, str]:
- """
- Get the information items from the lsb_release command output.
- Returns:
- A dictionary containing all information items.
- """
- if not self.include_lsb:
- return {}
- try:
- cmd = ("lsb_release", "-a")
- stdout = subprocess.check_output(cmd, stderr=subprocess.DEVNULL)
- # Command not found or lsb_release returned error
- except (OSError, subprocess.CalledProcessError):
- return {}
- content = self._to_str(stdout).splitlines()
- return self._parse_lsb_release_content(content)
- @staticmethod
- def _parse_lsb_release_content(lines: Iterable[str]) -> Dict[str, str]:
- """
- Parse the output of the lsb_release command.
- Parameters:
- * lines: Iterable through the lines of the lsb_release output.
- Each line must be a unicode string or a UTF-8 encoded byte
- string.
- Returns:
- A dictionary containing all information items.
- """
- props = {}
- for line in lines:
- kv = line.strip("\n").split(":", 1)
- if len(kv) != 2:
- # Ignore lines without colon.
- continue
- k, v = kv
- props.update({k.replace(" ", "_").lower(): v.strip()})
- return props
- @cached_property
- def _uname_info(self) -> Dict[str, str]:
- if not self.include_uname:
- return {}
- try:
- cmd = ("uname", "-rs")
- stdout = subprocess.check_output(cmd, stderr=subprocess.DEVNULL)
- except OSError:
- return {}
- content = self._to_str(stdout).splitlines()
- return self._parse_uname_content(content)
- @cached_property
- def _oslevel_info(self) -> str:
- if not self.include_oslevel:
- return ""
- try:
- stdout = subprocess.check_output("oslevel", stderr=subprocess.DEVNULL)
- except (OSError, subprocess.CalledProcessError):
- return ""
- return self._to_str(stdout).strip()
- @cached_property
- def _debian_version(self) -> str:
- try:
- with open(
- os.path.join(self.etc_dir, "debian_version"), encoding="ascii"
- ) as fp:
- return fp.readline().rstrip()
- except FileNotFoundError:
- return ""
- @staticmethod
- def _parse_uname_content(lines: Sequence[str]) -> Dict[str, str]:
- if not lines:
- return {}
- props = {}
- match = re.search(r"^([^\s]+)\s+([\d\.]+)", lines[0].strip())
- if match:
- name, version = match.groups()
- # This is to prevent the Linux kernel version from
- # appearing as the 'best' version on otherwise
- # identifiable distributions.
- if name == "Linux":
- return {}
- props["id"] = name.lower()
- props["name"] = name
- props["release"] = version
- return props
- @staticmethod
- def _to_str(bytestring: bytes) -> str:
- encoding = sys.getfilesystemencoding()
- return bytestring.decode(encoding)
- @cached_property
- def _distro_release_info(self) -> Dict[str, str]:
- """
- Get the information items from the specified distro release file.
- Returns:
- A dictionary containing all information items.
- """
- if self.distro_release_file:
- # If it was specified, we use it and parse what we can, even if
- # its file name or content does not match the expected pattern.
- distro_info = self._parse_distro_release_file(self.distro_release_file)
- basename = os.path.basename(self.distro_release_file)
- # The file name pattern for user-specified distro release files
- # is somewhat more tolerant (compared to when searching for the
- # file), because we want to use what was specified as best as
- # possible.
- match = _DISTRO_RELEASE_BASENAME_PATTERN.match(basename)
- else:
- try:
- basenames = [
- basename
- for basename in os.listdir(self.etc_dir)
- if basename not in _DISTRO_RELEASE_IGNORE_BASENAMES
- and os.path.isfile(os.path.join(self.etc_dir, basename))
- ]
- # We sort for repeatability in cases where there are multiple
- # distro specific files; e.g. CentOS, Oracle, Enterprise all
- # containing `redhat-release` on top of their own.
- basenames.sort()
- except OSError:
- # This may occur when /etc is not readable but we can't be
- # sure about the *-release files. Check common entries of
- # /etc for information. If they turn out to not be there the
- # error is handled in `_parse_distro_release_file()`.
- basenames = _DISTRO_RELEASE_BASENAMES
- for basename in basenames:
- match = _DISTRO_RELEASE_BASENAME_PATTERN.match(basename)
- if match is None:
- continue
- filepath = os.path.join(self.etc_dir, basename)
- distro_info = self._parse_distro_release_file(filepath)
- # The name is always present if the pattern matches.
- if "name" not in distro_info:
- continue
- self.distro_release_file = filepath
- break
- else: # the loop didn't "break": no candidate.
- return {}
- if match is not None:
- distro_info["id"] = match.group(1)
- # CloudLinux < 7: manually enrich info with proper id.
- if "cloudlinux" in distro_info.get("name", "").lower():
- distro_info["id"] = "cloudlinux"
- return distro_info
- def _parse_distro_release_file(self, filepath: str) -> Dict[str, str]:
- """
- Parse a distro release file.
- Parameters:
- * filepath: Path name of the distro release file.
- Returns:
- A dictionary containing all information items.
- """
- try:
- with open(filepath, encoding="utf-8") as fp:
- # Only parse the first line. For instance, on SLES there
- # are multiple lines. We don't want them...
- return self._parse_distro_release_content(fp.readline())
- except OSError:
- # Ignore not being able to read a specific, seemingly version
- # related file.
- # See https://github.com/python-distro/distro/issues/162
- return {}
- @staticmethod
- def _parse_distro_release_content(line: str) -> Dict[str, str]:
- """
- Parse a line from a distro release file.
- Parameters:
- * line: Line from the distro release file. Must be a unicode string
- or a UTF-8 encoded byte string.
- Returns:
- A dictionary containing all information items.
- """
- matches = _DISTRO_RELEASE_CONTENT_REVERSED_PATTERN.match(line.strip()[::-1])
- distro_info = {}
- if matches:
- # regexp ensures non-None
- distro_info["name"] = matches.group(3)[::-1]
- if matches.group(2):
- distro_info["version_id"] = matches.group(2)[::-1]
- if matches.group(1):
- distro_info["codename"] = matches.group(1)[::-1]
- elif line:
- distro_info["name"] = line.strip()
- return distro_info
- _distro = LinuxDistribution()
- def main() -> None:
- logger = logging.getLogger(__name__)
- logger.setLevel(logging.DEBUG)
- logger.addHandler(logging.StreamHandler(sys.stdout))
- parser = argparse.ArgumentParser(description="OS distro info tool")
- parser.add_argument(
- "--json", "-j", help="Output in machine readable format", action="store_true"
- )
- parser.add_argument(
- "--root-dir",
- "-r",
- type=str,
- dest="root_dir",
- help="Path to the root filesystem directory (defaults to /)",
- )
- args = parser.parse_args()
- if args.root_dir:
- dist = LinuxDistribution(
- include_lsb=False,
- include_uname=False,
- include_oslevel=False,
- root_dir=args.root_dir,
- )
- else:
- dist = _distro
- if args.json:
- logger.info(json.dumps(dist.info(), indent=4, sort_keys=True))
- else:
- logger.info("Name: %s", dist.name(pretty=True))
- distribution_version = dist.version(pretty=True)
- logger.info("Version: %s", distribution_version)
- distribution_codename = dist.codename()
- logger.info("Codename: %s", distribution_codename)
- if __name__ == "__main__":
- main()
|