Source code for extra_platforms

# Copyright Kevin Deldycke <kevin@deldycke.com> and contributors.
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
"""Expose package-wide elements."""

import sys
from functools import cache
from platform import platform
from typing import Callable, Iterable, Union

# This message is put up there because it is used in multiple places in other files.
_report_msg = (
    "Please report this at https://github.com/kdeldycke/extra-platforms/issues to "
    "improve detection heuristics."
)

from .detection import (  # noqa: E402
    is_aix,
    is_altlinux,
    is_amzn,
    is_android,
    is_arch,
    is_buildroot,
    is_centos,
    is_cloudlinux,
    is_cygwin,
    is_debian,
    is_exherbo,
    is_fedora,
    is_freebsd,
    is_gentoo,
    is_guix,
    is_hurd,
    is_ibm_powerkvm,
    is_kvmibm,
    is_linuxmint,
    is_macos,
    is_mageia,
    is_mandriva,
    is_midnightbsd,
    is_netbsd,
    is_nobara,
    is_openbsd,
    is_opensuse,
    is_oracle,
    is_parallels,
    is_pidora,
    is_raspbian,
    is_rhel,
    is_rocky,
    is_scientific,
    is_slackware,
    is_sles,
    is_solaris,
    is_sunos,
    is_tumbleweed,
    is_tuxedo,
    is_ubuntu,
    is_unknown_linux,
    is_windows,
    is_wsl1,
    is_wsl2,
    is_xenserver,
)
from .group import Group  # noqa: E402
from .group_data import (  # noqa: E402
    ALL_GROUPS,
    ALL_PLATFORMS,
    ANY_WINDOWS,
    BSD,
    BSD_WITHOUT_MACOS,
    EXTRA_GROUPS,
    LINUX,
    LINUX_LAYERS,
    LINUX_LIKE,
    NON_OVERLAPPING_GROUPS,
    OTHER_UNIX,
    SYSTEM_V,
    UNIX,
    UNIX_LAYERS,
    UNIX_WITHOUT_MACOS,
)
from .platform import Platform  # noqa: E402
from .platform_data import (  # noqa: E402
    AIX,
    ALTLINUX,
    AMZN,
    ANDROID,
    ARCH,
    BUILDROOT,
    CENTOS,
    CLOUDLINUX,
    CYGWIN,
    DEBIAN,
    EXHERBO,
    FEDORA,
    FREEBSD,
    GENTOO,
    GUIX,
    HURD,
    IBM_POWERKVM,
    KVMIBM,
    LINUXMINT,
    MACOS,
    MAGEIA,
    MANDRIVA,
    MIDNIGHTBSD,
    NETBSD,
    NOBARA,
    OPENBSD,
    OPENSUSE,
    ORACLE,
    PARALLELS,
    PIDORA,
    RASPBIAN,
    RHEL,
    ROCKY,
    SCIENTIFIC,
    SLACKWARE,
    SLES,
    SOLARIS,
    SUNOS,
    TUMBLEWEED,
    TUXEDO,
    UBUNTU,
    UNKNOWN_LINUX,
    WINDOWS,
    WSL1,
    WSL2,
    XENSERVER,
)

"""
.. important::
    Exposing everything at package level here motivates platforms and groups to have a
    unique and unambiguous ID. This constraint is enforced at the data-level and
    checked in unittests.

.. caution::
    The content of ``pytest.py`` file is not imported here at package level like this:

    .. code-block:: python
        from .pytest import *

    That is to make dependency on Pytest optional.
"""


_TPlatformRef = Platform | Group | str | None
"""All types that can be used to reference a platform or a group:

- a ``Platform`` object itself
- a ``Group`` object representing a collection of platforms
- a string representing a platform ID or a group ID
- ``None`` to represent an empty set of platforms
"""

_TNestedReferences = Union[
    _TPlatformRef | Iterable[_TPlatformRef | Iterable["_TNestedReferences"]]
]
"""Type for arbitrary nested references to platforms and groups."""


from .operations import (  # noqa: E402
    ALL_GROUP_IDS,
    ALL_IDS,
    ALL_PLATFORM_IDS,
    groups_from_ids,
    platforms_from_ids,
    reduce,
)

__version__ = "3.1.1"
"""Examples of valid version strings according :pep:`440#version-scheme`:

.. code-block:: python

    __version__ = "1.2.3.dev1"  # Development release 1
    __version__ = "1.2.3a1"  # Alpha Release 1
    __version__ = "1.2.3b1"  # Beta Release 1
    __version__ = "1.2.3rc1"  # RC Release 1
    __version__ = "1.2.3"  # Final Release
    __version__ = "1.2.3.post1"  # Post Release 1
"""


[docs] @cache def current_platforms() -> tuple[Platform, ...]: """Evaluates all heuristics and returns a list of ``Platform`` matching the current environment. Always raises an error if the current environment is not recognized. ..attention:: At this point it is too late to worry about caching. This function has no choice but to evaluate all platforms detection heuristics. """ matching = [] for p in ALL_PLATFORMS.platforms: if p.current: matching.append(p) if not matching: msg = ( f"Unrecognized {sys.platform} / " f"{platform(aliased=True, terse=True)} platform. {_report_msg}" ) raise SystemError(msg) return tuple(matching)
[docs] @cache def current_os() -> Platform: """Always returns the best matching platform for the current environment. If multiple platforms match the current environment, this function will try to select the best, informative one. Raises an error if we can't decide on a single, appropriate platform. """ matching = set(current_platforms()) # Return the only matching platform. if len(matching) == 1: return matching.pop() # Remove unknown Linux, which is too generic to be useful. if UNKNOWN_LINUX in matching: matching.remove(UNKNOWN_LINUX) if len(matching) == 1: return matching.pop() # Remove WSL1, then WSL2, until we have a single match. WSL is a generic platform, # so we should prefer the other, more specific platform matches like Ubuntu. See: # - https://github.com/kdeldycke/extra-platforms/issues/158 # - https://github.com/kdeldycke/meta-package-manager/issues/944 for wsl in (WSL1, WSL2): if wsl in matching: matching.remove(wsl) if len(matching) == 1: return matching.pop() # Our meta-heuristics above failed to decide on a single, appropriate platform. msg = f"Multiple platforms match current environment: {matching} . {_report_msg}" raise RuntimeError(msg)
def _generate_group_membership_func(_group: Group) -> Callable: """Factory to produce dynamiccaly a group membership test function.""" def group_membership_check() -> bool: """Evaluates membership of the current platform to the group. Returns ``True`` if the current platform is part of the group, ``False`` otherwise. """ return current_os() in _group group_membership_check.__doc__ = ( "Returns ``True`` if the current platform is part of the group composed of " f"{_group.short_desc}, ``False`` otherwise." ) return cache(group_membership_check) for _group in ALL_GROUPS: func_id = f"is_{_group.id}" assert func_id not in locals(), f"Function ID {func_id} already defined locally." globals()[func_id] = _generate_group_membership_func(_group) """Generates an ``is_<group.id>()`` local function for each group. These functions return a boolean value indicating the membership of the current platform into that group. Since platforms and groups have unique, non-overlapping IDs, we can create a ``is_<group.id>`` method for each group. The value of this boolean variable mark the membership of the current platform to that group. """ __all__ = ( "AIX", # noqa: F405 "ALL_GROUP_IDS", # noqa: F405 "ALL_GROUPS", # noqa: F405 "ALL_IDS", # noqa: F405 "ALL_PLATFORM_IDS", # noqa: F405 "ALL_PLATFORMS", # noqa: F405 "ALTLINUX", # noqa: F405 "AMZN", # noqa: F405 "ANDROID", # noqa: F405 "ANY_WINDOWS", # noqa: F405 "ARCH", # noqa: F405 "BSD", # noqa: F405 "BSD_WITHOUT_MACOS", # noqa: F405 "BUILDROOT", # noqa: F405 "CENTOS", # noqa: F405 "CLOUDLINUX", # noqa: F405 "current_os", # noqa: F405 "current_platforms", # noqa: F405 "CYGWIN", # noqa: F405 "DEBIAN", # noqa: F405 "EXHERBO", # noqa: F405 "EXTRA_GROUPS", # noqa: F405 "FEDORA", # noqa: F405 "FREEBSD", # noqa: F405 "GENTOO", # noqa: F405 "Group", # noqa: F405 "groups_from_ids", # noqa: F405 "GUIX", # noqa: F405 "HURD", # noqa: F405 "IBM_POWERKVM", # noqa: F405 "is_aix", # noqa: F405 "is_all_platforms", # noqa: F405, F822 "is_altlinux", # noqa: F405 "is_amzn", # noqa: F405 "is_android", # noqa: F405 "is_any_windows", # noqa: F405, F822 "is_arch", # noqa: F405 "is_bsd", # noqa: F405, F822 "is_bsd_without_macos", # noqa: F405, F822 "is_buildroot", # noqa: F405 "is_centos", # noqa: F405 "is_cloudlinux", # noqa: F405 "is_cygwin", # noqa: F405 "is_debian", # noqa: F405 "is_exherbo", # noqa: F405 "is_fedora", # noqa: F405 "is_freebsd", # noqa: F405 "is_gentoo", # noqa: F405 "is_guix", # noqa: F405 "is_hurd", # noqa: F405 "is_ibm_powerkvm", # noqa: F405 "is_kvmibm", # noqa: F405 "is_linux", # noqa: F405, F822 "is_linux_layers", # noqa: F405, F822 "is_linux_like", # noqa: F405, F822 "is_linuxmint", # noqa: F405 "is_macos", # noqa: F405 "is_mageia", # noqa: F405 "is_mandriva", # noqa: F405 "is_midnightbsd", # noqa: F405 "is_netbsd", # noqa: F405 "is_nobara", # noqa: F405 "is_openbsd", # noqa: F405 "is_opensuse", # noqa: F405 "is_oracle", # noqa: F405 "is_other_unix", # noqa: F405, F822 "is_parallels", # noqa: F405 "is_pidora", # noqa: F405 "is_raspbian", # noqa: F405 "is_rhel", # noqa: F405 "is_rocky", # noqa: F405 "is_scientific", # noqa: F405 "is_slackware", # noqa: F405 "is_sles", # noqa: F405 "is_solaris", # noqa: F405 "is_sunos", # noqa: F405 "is_system_v", # noqa: F405, F822 "is_tumbleweed", # noqa: F405, F822 "is_tuxedo", # noqa: F405, F822 "is_ubuntu", # noqa: F405 "is_unix", # noqa: F405, F822 "is_unix_layers", # noqa: F405, F822 "is_unix_without_macos", # noqa: F405, F822 "is_unknown_linux", # noqa: F405 "is_windows", # noqa: F405 "is_wsl1", # noqa: F405 "is_wsl2", # noqa: F405 "is_xenserver", # noqa: F405 "KVMIBM", # noqa: F405 "LINUX", # noqa: F405 "LINUX_LAYERS", # noqa: F405 "LINUX_LIKE", # noqa: F405 "LINUXMINT", # noqa: F405 "MACOS", # noqa: F405 "MAGEIA", # noqa: F405 "MANDRIVA", # noqa: F405 "MIDNIGHTBSD", # noqa: F405 "NETBSD", # noqa: F405 "NOBARA", # noqa: F405 "NON_OVERLAPPING_GROUPS", # noqa: F405 "OPENBSD", # noqa: F405 "OPENSUSE", # noqa: F405 "ORACLE", # noqa: F405 "OTHER_UNIX", # noqa: F405 "PARALLELS", # noqa: F405 "PIDORA", # noqa: F405 "Platform", # noqa: F405 "platforms_from_ids", # noqa: F405 "RASPBIAN", # noqa: F405 "reduce", # noqa: F405 "RHEL", # noqa: F405 "ROCKY", # noqa: F405 "SCIENTIFIC", # noqa: F405 "SLACKWARE", # noqa: F405 "SLES", # noqa: F405 "SOLARIS", # noqa: F405 "SUNOS", # noqa: F405 "SYSTEM_V", # noqa: F405 "TUMBLEWEED", # noqa: F405 "TUXEDO", # noqa: F405 "UBUNTU", # noqa: F405 "UNIX", # noqa: F405 "UNIX_LAYERS", # noqa: F405 "UNIX_WITHOUT_MACOS", # noqa: F405 "UNKNOWN_LINUX", # noqa: F405 "WINDOWS", # noqa: F405 "WSL1", # noqa: F405 "WSL2", # noqa: F405 "XENSERVER", # noqa: F405 ) """Expose all package-wide elements. .. note:: The content of ``__all__`` is checked and enforced in unittests. .. todo:: Test Ruff's ``__all__`` formatting capabilities. And if good enough, remove ``__all__`` checks in unittests. """