Groups¶
Group usage¶
Todo
Explain high-level usage of groups here. Including membership testing and set operations. And how to create custom groups (from scratch or by combining existing groups).
All groups¶
All recognized groups and their properties:
Icon |
Symbol |
Description |
||
|---|---|---|---|---|
🏛️ |
All architectures |
|||
📱 |
ARM architectures |
⬥ |
||
♺ |
CI systems |
⬥ |
||
🔲 |
MIPS architectures |
⬥ |
||
⚙️ |
All platforms |
|||
☀️ |
SPARC architectures |
⬥ |
||
⁕ |
All architectures, platforms and CI systems |
|||
🪟 |
All Windows |
⬥ |
||
³² |
32-bit architectures |
|||
⁶⁴ |
64-bit architectures |
|||
🅱️+ |
All BSD |
⬥ |
||
🅱️ |
All BSD excluding macOS |
|||
🏢 |
IBM mainframe |
⬥ |
||
🐧 |
Linux distributions |
⬥ |
||
≚ |
Linux compatibility layers |
⬥ |
||
🐧+ |
All Linux & compatibility layers |
|||
🐉 |
LoongArch |
⬥ |
||
🅟 |
Other POSIX-compliant platforms |
⬥ |
||
⚡ |
PowerPC family |
⬥ |
||
Ⅴ |
RISC-V family |
⬥ |
||
𝐕 |
AT&T System Five |
⬥ |
||
⨷ |
All Unix |
|||
≛ |
Unix compatibility layers |
⬥ |
||
⨂ |
All Unix excluding macOS |
|||
❓ |
Unknown |
⬥ |
||
🌐 |
WebAssembly |
⬥ |
||
𝘅 |
x86 family |
⬥ |
Hint
Canonical groups are non-overlapping groups that together cover all recognized traits. They are marked with a ⬥ icon in the table above.
Other groups are provided for convenience, but overlap with each other or with canonical groups.
Predefined groups¶
- extra_platforms.ALL_ARCHITECTURES = Group(id='all_architectures', name='All architectures')¶
A
Groupidentifies a collection ofTraitmembers.Additionally of the common fields inherited from
_Identifiable, each group provides:members: An iterable ofTraitinstances that belong to this group.member_ids: Afrozensetof member IDs for quick lookup.canonical: Aboolindicating if the group is canonical (non-overlapping).various
set-like operations (union, intersection, difference, etc.).ID:
all_architecturesName: All architectures
Icon: 🏛️
Canonical:
FalseDetection function:
is_any_architecture()Pytest decorators:
skip_all_architectures/unless_any_architectureMembers (25
Architecture):AARCH64,ARM,ARMV5TEL,ARMV6L,ARMV7L,ARMV8L,I386,I586,I686,LOONGARCH64,MIPS,MIPS64,MIPS64EL,MIPSEL,PPC,PPC64,PPC64LE,RISCV32,RISCV64,S390X,SPARC,SPARC64,WASM32,WASM64,X86_64
- extra_platforms.ALL_ARM = Group(id='all_arm', name='ARM architectures')¶
All ARM-based architectures.
ID:
all_armName: ARM architectures
Icon: 📱
Canonical:
True⬥Detection function:
is_any_arm()Pytest decorators:
skip_all_arm/unless_any_armMembers (6
Architecture):AARCH64,ARM,ARMV5TEL,ARMV6L,ARMV7L,ARMV8L
- extra_platforms.ALL_CI = Group(id='all_ci', name='CI systems')¶
All recognized Continuous Integration systems.
Caution
This group does not contain the
UNKNOWN_CItrait.See also
ID:
all_ciName: CI systems
Icon: ♺
Canonical:
True⬥Detection function:
is_any_ci()Pytest decorators:
skip_all_ci/unless_any_ciMembers (11
CI):AZURE_PIPELINES,BAMBOO,BUILDKITE,CIRCLE_CI,CIRRUS_CI,CODEBUILD,GITHUB_CI,GITLAB_CI,HEROKU_CI,TEAMCITY,TRAVIS_CI
- extra_platforms.ALL_MIPS = Group(id='all_mips', name='MIPS architectures')¶
All MIPS-based architectures.
ID:
all_mipsName: MIPS architectures
Icon: 🔲
Canonical:
True⬥Detection function:
is_any_mips()Pytest decorators:
skip_all_mips/unless_any_mipsMembers (4
Architecture):MIPS,MIPS64,MIPS64EL,MIPSEL
- extra_platforms.ALL_PLATFORMS = Group(id='all_platforms', name='All platforms')¶
A
Groupidentifies a collection ofTraitmembers.Additionally of the common fields inherited from
_Identifiable, each group provides:members: An iterable ofTraitinstances that belong to this group.member_ids: Afrozensetof member IDs for quick lookup.canonical: Aboolindicating if the group is canonical (non-overlapping).various
set-like operations (union, intersection, difference, etc.).ID:
all_platformsName: All platforms
Icon: ⚙️
Canonical:
FalseDetection function:
is_any_platform()Pytest decorators:
skip_all_platforms/unless_any_platformMembers (50
Platform):AIX,ALTLINUX,AMZN,ANDROID,ARCH,BUILDROOT,CACHYOS,CENTOS,CLOUDLINUX,CYGWIN,DEBIAN,DRAGONFLY_BSD,EXHERBO,FEDORA,FREEBSD,GENTOO,GUIX,HAIKU,HURD,IBM_POWERKVM,ILLUMOS,KVMIBM,LINUXMINT,MACOS,MAGEIA,MANDRIVA,MIDNIGHTBSD,NETBSD,NOBARA,OPENBSD,OPENSUSE,ORACLE,PARALLELS,PIDORA,RASPBIAN,RHEL,ROCKY,SCIENTIFIC,SLACKWARE,SLES,SOLARIS,SUNOS,TUMBLEWEED,TUXEDO,UBUNTU,ULTRAMARINE,WINDOWS,WSL1,WSL2,XENSERVER
- extra_platforms.ALL_SPARC = Group(id='all_sparc', name='SPARC architectures')¶
All SPARC-based architectures.
ID:
all_sparcName: SPARC architectures
Icon: ☀️
Canonical:
True⬥Detection function:
is_any_sparc()Pytest decorators:
skip_all_sparc/unless_any_sparcMembers (2
Architecture):SPARC,SPARC64
- extra_platforms.ALL_TRAITS = Group(id='all_traits', name='All architectures, platforms and CI systems')¶
All predefined architectures, platforms and CI systems.
Hint
This group includes all
UNKNOWN_*traits.ID:
all_traitsName: All architectures, platforms and CI systems
Icon: ⁕
Canonical:
FalseDetection function:
is_any_trait()Pytest decorators:
skip_all_traits/unless_any_traitMembers (26
Architecture, 12CI, 51Platform):AARCH64,AIX,ALTLINUX,AMZN,ANDROID,ARCH,ARM,ARMV5TEL,ARMV6L,ARMV7L,ARMV8L,AZURE_PIPELINES,BAMBOO,BUILDKITE,BUILDROOT,CACHYOS,CENTOS,CIRCLE_CI,CIRRUS_CI,CLOUDLINUX,CODEBUILD,CYGWIN,DEBIAN,DRAGONFLY_BSD,EXHERBO,FEDORA,FREEBSD,GENTOO,GITHUB_CI,GITLAB_CI,GUIX,HAIKU,HEROKU_CI,HURD,I386,I586,I686,IBM_POWERKVM,ILLUMOS,KVMIBM,LINUXMINT,LOONGARCH64,MACOS,MAGEIA,MANDRIVA,MIDNIGHTBSD,MIPS,MIPS64,MIPS64EL,MIPSEL,NETBSD,NOBARA,OPENBSD,OPENSUSE,ORACLE,PARALLELS,PIDORA,PPC,PPC64,PPC64LE,RASPBIAN,RHEL,RISCV32,RISCV64,ROCKY,S390X,SCIENTIFIC,SLACKWARE,SLES,SOLARIS,SPARC,SPARC64,SUNOS,TEAMCITY,TRAVIS_CI,TUMBLEWEED,TUXEDO,UBUNTU,ULTRAMARINE,UNKNOWN_ARCHITECTURE,UNKNOWN_CI,UNKNOWN_PLATFORM,WASM32,WASM64,WINDOWS,WSL1,WSL2,X86_64,XENSERVER
- extra_platforms.ALL_WINDOWS = Group(id='all_windows', name='All Windows')¶
All Windows operating systems.
ID:
all_windowsName: All Windows
Icon: 🪟
Canonical:
True⬥Detection function:
is_any_windows()Pytest decorators:
skip_all_windows/unless_any_windows
- extra_platforms.ARCH_32_BIT = Group(id='arch_32_bit', name='32-bit architectures')¶
All 32-bit architectures.
ID:
arch_32_bitName: 32-bit architectures
Icon: ³²
Canonical:
FalseDetection function:
is_arch_32_bit()Pytest decorators:
skip_arch_32_bit/unless_arch_32_bitMembers (14
Architecture):ARM,ARMV5TEL,ARMV6L,ARMV7L,ARMV8L,I386,I586,I686,MIPS,MIPSEL,PPC,RISCV32,SPARC,WASM32
- extra_platforms.ARCH_64_BIT = Group(id='arch_64_bit', name='64-bit architectures')¶
All 64-bit architectures.
ID:
arch_64_bitName: 64-bit architectures
Icon: ⁶⁴
Canonical:
FalseDetection function:
is_arch_64_bit()Pytest decorators:
skip_arch_64_bit/unless_arch_64_bitMembers (11
Architecture):AARCH64,LOONGARCH64,MIPS64,MIPS64EL,PPC64,PPC64LE,RISCV64,S390X,SPARC64,WASM64,X86_64
- extra_platforms.BSD = Group(id='bsd', name='All BSD')¶
All BSD platforms.
Note
Are considered of this family (according Wikipedia):
386BSD (FreeBSD, NetBSD, OpenBSD, DragonFly BSD)
NeXTSTEP
Darwin (macOS, iOS, audioOS, iPadOS, tvOS, watchOS, bridgeOS)
SunOS
Ultrix
ID:
bsdName: All BSD
Icon: 🅱️+
Canonical:
True⬥Detection function:
is_bsd()Pytest decorators:
skip_bsd/unless_bsdMembers (7
Platform):DRAGONFLY_BSD,FREEBSD,MACOS,MIDNIGHTBSD,NETBSD,OPENBSD,SUNOS
- extra_platforms.BSD_WITHOUT_MACOS = Group(id='bsd_without_macos', name='All BSD excluding macOS')¶
All BSD platforms, without macOS.
This is useful to avoid macOS-specific workarounds on BSD platforms.
ID:
bsd_without_macosName: All BSD excluding macOS
Icon: 🅱️
Canonical:
FalseDetection function:
is_bsd_not_macos()Pytest decorators:
skip_bsd_not_macos/unless_bsd_not_macosMembers (6
Platform):DRAGONFLY_BSD,FREEBSD,MIDNIGHTBSD,NETBSD,OPENBSD,SUNOS
- extra_platforms.IBM_MAINFRAME = Group(id='ibm_mainframe', name='IBM mainframe')¶
IBM mainframe architectures.
ID:
ibm_mainframeName: IBM mainframe
Icon: 🏢
Canonical:
True⬥Detection function:
is_ibm_mainframe()Pytest decorators:
skip_ibm_mainframe/unless_ibm_mainframeMembers (1
Architecture):S390X
- extra_platforms.LINUX = Group(id='linux', name='Linux distributions')¶
All distributions based on a Linux kernel.
ID:
linuxName: Linux distributions
Icon: 🐧
Canonical:
True⬥Detection function:
is_linux()Pytest decorators:
skip_linux/unless_linuxMembers (34
Platform):ALTLINUX,AMZN,ANDROID,ARCH,BUILDROOT,CACHYOS,CENTOS,CLOUDLINUX,DEBIAN,EXHERBO,FEDORA,GENTOO,GUIX,IBM_POWERKVM,KVMIBM,LINUXMINT,MAGEIA,MANDRIVA,NOBARA,OPENSUSE,ORACLE,PARALLELS,PIDORA,RASPBIAN,RHEL,ROCKY,SCIENTIFIC,SLACKWARE,SLES,TUMBLEWEED,TUXEDO,UBUNTU,ULTRAMARINE,XENSERVER
- extra_platforms.LINUX_LAYERS = Group(id='linux_layers', name='Linux compatibility layers')¶
Interfaces that allows Linux binaries to run on a different host system.
ID:
linux_layersName: Linux compatibility layers
Icon: ≚
Canonical:
True⬥Detection function:
is_linux_layers()Pytest decorators:
skip_linux_layers/unless_linux_layers
- extra_platforms.LINUX_LIKE = Group(id='linux_like', name='All Linux & compatibility layers')¶
Sum of all Linux distributions and Linux compatibility layers.
ID:
linux_likeName: All Linux & compatibility layers
Icon: 🐧+
Canonical:
FalseDetection function:
is_linux_like()Pytest decorators:
skip_linux_like/unless_linux_likeMembers (36
Platform):ALTLINUX,AMZN,ANDROID,ARCH,BUILDROOT,CACHYOS,CENTOS,CLOUDLINUX,DEBIAN,EXHERBO,FEDORA,GENTOO,GUIX,IBM_POWERKVM,KVMIBM,LINUXMINT,MAGEIA,MANDRIVA,NOBARA,OPENSUSE,ORACLE,PARALLELS,PIDORA,RASPBIAN,RHEL,ROCKY,SCIENTIFIC,SLACKWARE,SLES,TUMBLEWEED,TUXEDO,UBUNTU,ULTRAMARINE,WSL1,WSL2,XENSERVER
- extra_platforms.LOONGARCH = Group(id='loongarch', name='LoongArch')¶
LoongArch architecture.
ID:
loongarchName: LoongArch
Icon: 🐉
Canonical:
True⬥Detection function:
is_loongarch()Pytest decorators:
skip_loongarch/unless_loongarchMembers (1
Architecture):LOONGARCH64
- extra_platforms.OTHER_POSIX = Group(id='other_posix', name='Other POSIX-compliant platforms')¶
All other UNIX-like or POSIX-compliant platforms.
Note
Are considered of this family (according Wikipedia):
Coherent
GNU/Hurd
HarmonyOS
LiteOS
LynxOS
Minix
MOS
OSF/1
QNX
BlackBerry 10
Research Unix
SerenityOS
ID:
other_posixName: Other POSIX-compliant platforms
Icon: 🅟
Canonical:
True⬥Detection function:
is_other_posix()Pytest decorators:
skip_other_posix/unless_other_posix
- extra_platforms.POWERPC = Group(id='powerpc', name='PowerPC family')¶
All PowerPC-based architectures.
ID:
powerpcName: PowerPC family
Icon: ⚡
Canonical:
True⬥Detection function:
is_powerpc()Pytest decorators:
skip_powerpc/unless_powerpcMembers (3
Architecture):PPC,PPC64,PPC64LE
- extra_platforms.RISCV = Group(id='riscv', name='RISC-V family')¶
All RISC-V-based architectures.
ID:
riscvName: RISC-V family
Icon: Ⅴ
Canonical:
True⬥Detection function:
is_riscv()Pytest decorators:
skip_riscv/unless_riscvMembers (2
Architecture):RISCV32,RISCV64
- extra_platforms.SYSTEM_V = Group(id='system_v', name='AT&T System Five')¶
All Unix platforms derived from AT&T System Five.
Note
Are considered of this family (according Wikipedia):
A/UX
AIX
HP-UX
IRIX
OpenServer
Solaris
OpenSolaris
Illumos
Tru64
UNIX
UnixWare
ID:
system_vName: AT&T System Five
Icon: 𝐕
Canonical:
True⬥Detection function:
is_system_v()Pytest decorators:
skip_system_v/unless_system_v
- extra_platforms.UNIX = Group(id='unix', name='All Unix')¶
All Unix-like operating systems and compatibility layers.
ID:
unixName: All Unix
Icon: ⨷
Canonical:
FalseDetection function:
is_unix()Pytest decorators:
skip_unix/unless_unixMembers (49
Platform):AIX,ALTLINUX,AMZN,ANDROID,ARCH,BUILDROOT,CACHYOS,CENTOS,CLOUDLINUX,CYGWIN,DEBIAN,DRAGONFLY_BSD,EXHERBO,FEDORA,FREEBSD,GENTOO,GUIX,HAIKU,HURD,IBM_POWERKVM,ILLUMOS,KVMIBM,LINUXMINT,MACOS,MAGEIA,MANDRIVA,MIDNIGHTBSD,NETBSD,NOBARA,OPENBSD,OPENSUSE,ORACLE,PARALLELS,PIDORA,RASPBIAN,RHEL,ROCKY,SCIENTIFIC,SLACKWARE,SLES,SOLARIS,SUNOS,TUMBLEWEED,TUXEDO,UBUNTU,ULTRAMARINE,WSL1,WSL2,XENSERVER
- extra_platforms.UNIX_LAYERS = Group(id='unix_layers', name='Unix compatibility layers')¶
Interfaces that allows Unix binaries to run on a different host system.
Note
Are considered of this family (according Wikipedia):
Cygwin
Darling
Eunice
GNV
Interix
MachTen
Microsoft POSIX subsystem
MKS Toolkit
PASE
P.I.P.S.
PWS/VSE-AF
UNIX System Services
UserLAnd Technologies
Windows Services for UNIX
ID:
unix_layersName: Unix compatibility layers
Icon: ≛
Canonical:
True⬥Detection function:
is_unix_layers()Pytest decorators:
skip_unix_layers/unless_unix_layers
- extra_platforms.UNIX_WITHOUT_MACOS = Group(id='unix_without_macos', name='All Unix excluding macOS')¶
All Unix platforms, without macOS.
This is useful to avoid macOS-specific workarounds on Unix platforms.
ID:
unix_without_macosName: All Unix excluding macOS
Icon: ⨂
Canonical:
FalseDetection function:
is_unix_not_macos()Pytest decorators:
skip_unix_not_macos/unless_unix_not_macosMembers (48
Platform):AIX,ALTLINUX,AMZN,ANDROID,ARCH,BUILDROOT,CACHYOS,CENTOS,CLOUDLINUX,CYGWIN,DEBIAN,DRAGONFLY_BSD,EXHERBO,FEDORA,FREEBSD,GENTOO,GUIX,HAIKU,HURD,IBM_POWERKVM,ILLUMOS,KVMIBM,LINUXMINT,MAGEIA,MANDRIVA,MIDNIGHTBSD,NETBSD,NOBARA,OPENBSD,OPENSUSE,ORACLE,PARALLELS,PIDORA,RASPBIAN,RHEL,ROCKY,SCIENTIFIC,SLACKWARE,SLES,SOLARIS,SUNOS,TUMBLEWEED,TUXEDO,UBUNTU,ULTRAMARINE,WSL1,WSL2,XENSERVER
- extra_platforms.UNKNOWN = Group(id='unknown', name='Unknown')¶
Unknown or unrecognized traits.
ID:
unknownName: Unknown
Icon: ❓
Canonical:
True⬥Detection function:
is_unknown()Pytest decorators:
skip_unknown/unless_unknownMembers (1
Architecture, 1CI, 1Platform):UNKNOWN_ARCHITECTURE,UNKNOWN_CI,UNKNOWN_PLATFORM
- extra_platforms.WEBASSEMBLY = Group(id='webassembly', name='WebAssembly')¶
WebAssembly architectures.
ID:
webassemblyName: WebAssembly
Icon: 🌐
Canonical:
True⬥Detection function:
is_webassembly()Pytest decorators:
skip_webassembly/unless_webassemblyMembers (2
Architecture):WASM32,WASM64
- extra_platforms.X86 = Group(id='x86', name='x86 family')¶
All x86-based architectures (Intel-compatible).
ID:
x86Name: x86 family
Icon: 𝘅
Canonical:
True⬥Detection function:
is_x86()Pytest decorators:
skip_x86/unless_x86Members (4
Architecture):I386,I586,I686,X86_64
Todo
List and document group collections: ALL_GROUP_IDS, ALL_TRAIT_IDS, ALL_IDS, …
Group implementation¶
classDiagram
_Identifiable <|-- Group
Group a collection of traits. Also referred as families.
- class extra_platforms.group.Group(id, name, icon='❓', members=<factory>)[source]¶
Bases:
_IdentifiableA
Groupidentifies a collection ofTraitmembers.Additionally of the common fields inherited from
_Identifiable, each group provides:members: An iterable ofTraitinstances that belong to this group.member_ids: Afrozensetof member IDs for quick lookup.canonical: Aboolindicating if the group is canonical (non-overlapping).various
set-like operations (union, intersection, difference, etc.).
- unknown_symbol: ClassVar[str] = 'UNKNOWN'¶
Groups use
UNKNOWNinstead ofUNKNOWN_GROUP.
- members: Iterable[Trait]¶
Traits in this group.
Normalized to
MappingProxyType[str, Trait]at init, providing O(1) lookup by ID.
- property canonical: bool¶
Returns True if the group is canonical (non-overlapping), False otherwise.
A canonical group is one that does not share any members with other canonical groups. All canonical groups are non-overlapping.
Non-canonical groups are provided for convenience, but overlap with each other or with canonical groups.
Hint
Canonical groups are denoted with a ⬥ symbol in the documentation and tables.
- isdisjoint(other)[source]¶
Return True if the group has no members in common with
other.Groups are disjoint if and only if their intersection is an empty set.
othercan be an arbitrarily nestedIterableofGroupandTrait.- Return type:
- fullyintersects(other)[source]¶
Return True if the group has all members in common with
other.- Return type:
- union(*others)[source]¶
Return a new
Groupwith members from the group and all others.Caution
The new
Groupwill inherits the metadata of the first one. All other groups’ metadata will be ignored.- Return type:
- intersection(*others)[source]¶
Return a new
Groupwith members common to the group and all others.Caution
The new
Groupwill inherits the metadata of the first one. All other groups’ metadata will be ignored.- Return type:
- difference(*others)[source]¶
Return a new
Groupwith members in the group that are not in the others.Caution
The new
Groupwill inherits the metadata of the first one. All other groups’ metadata will be ignored.- Return type:
- symmetric_difference(other)[source]¶
Return a new
Groupwith members in either the group or other but not both.Caution
The new
Groupwill inherits the metadata of the first one. All other groups’ metadata will be ignored.- Return type:
- copy(id=None, name=None, icon=None, members=None)[source]¶
Return a shallow copy of the group.
Fields can be overridden by passing new values as arguments.
- Return type:
- add(member)[source]¶
Return a new
Groupwith the specified trait added.If the trait is already in the group, returns a copy unchanged.
- Return type:
- Args:
member: A
Traitobject or trait ID string to add.- Returns:
A new
Groupinstance with the trait added.- Raises:
ValueError: If the trait ID is not recognized.
- remove(member)[source]¶
Return a new
Groupwith the specified trait removed.Raises
KeyErrorif the trait is not in the group.- Return type:
- Args:
member: A
Traitobject or trait ID string to remove.- Returns:
A new
Groupinstance with the trait removed.- Raises:
KeyError: If the trait is not in the group.
- discard(member)[source]¶
Return a new
Groupwith the specified trait removed if present.Unlike
remove(), this does not raise an error if the trait is not found.- Return type:
- Args:
member: A
Traitobject or trait ID string to remove.- Returns:
A new
Groupinstance with the trait removed, or a copy if not present.
- all_group: ClassVar[str] = 'ALL_GROUPS'¶
The symbol name for the group containing all instances of this type.
- data_module_id: ClassVar[str] = 'group_data'¶
The module name where instances of this type are defined.
- doc_page: ClassVar[str] = 'groups.md'¶
The documentation page filename.
- pop(member_id=None)[source]¶
Remove and return a trait from the group.
- Args:
- member_id: Optional trait ID to remove. If not provided, removes an arbitrary
trait (specifically, the first one in iteration order).
- Returns:
A tuple of (removed_trait, new_group).
- Raises:
KeyError: If
member_idis provided but not found in the group. KeyError: If the group is empty.
- type_id: ClassVar[str] = 'group'¶
Machine-readable type identifier used to derive module and symbol names.
- type_name: ClassVar[str] = 'group'¶
Human-readable type name for documentation.
- id: str¶
Unique ID of the object.
- symbol_id: str¶
Symbolic identifier.
This is the variable name under which the instance can be accessed at the root of the
extra_platformsmodule.Mainly useful for documentation generation.
- detection_func_id: str¶
ID of the detection function for this object.
The detection function is expected to be named
is_<id>()and available at the root of theextra_platformsmodule.
- skip_decorator_id: str¶
ID of the Pytest skip decorator for this object.
The decorator is expected to be named
@skip_<id>and available from theextra_platforms.pytestmodule.
- unless_decorator_id: str¶
ID of the Pytest unless decorator for this object.
The decorator is expected to be named
@unless_<id>and available from theextra_platforms.pytestmodule.
- name: str¶
User-friendly name of the object.
Definitions of ready-to-use groups.