# Copyright (c) 2015 SUSE Linux GmbH. All rights reserved.
#
# This file is part of kiwi.
#
# kiwi 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 3 of the License, or
# (at your option) any later version.
#
# kiwi 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 kiwi. If not, see <http://www.gnu.org/licenses/>
#
import logging
import os
import glob
import importlib
from importlib.resources import as_file
from importlib.util import find_spec
from importlib.machinery import ModuleSpec
from collections import namedtuple
import platform
import yaml
from typing import (
List, NamedTuple, Optional, Dict
)
# project
from kiwi.path import Path
from kiwi.version import (
__githash__,
__version__
)
from kiwi.exceptions import KiwiBootLoaderGrubDataError
shim_loader_type = NamedTuple(
'shim_loader_type', [
('filename', str),
('binaryname', str)
]
)
grub_loader_type = NamedTuple(
'grub_loader_type', [
('filename', str),
('binaryname', str)
]
)
unit_type = NamedTuple(
'unit_type', [
('byte', str),
('kb', str),
('mb', str),
('gb', str)
]
)
# Default module variables
DM_METADATA_FORMAT_VERSION = '1'
DM_METADATA_OFFSET = 4096 # 4kb
VERITY_DATA_BLOCKSIZE = 4096 # 4kb
VERITY_HASH_BLOCKSIZE = 4096 # 4kb
INTEGRITY_SECTOR_SIZE = 512
INTEGRITY_ALGORITHM = 'sha256'
INTEGRITY_KEY_ALGORITHM = 'hmac-sha256'
UNIT = unit_type(byte='b', kb='k', mb='m', gb='g')
POST_DISK_SYNC_SCRIPT = 'disk.sh'
PRE_DISK_SYNC_SCRIPT = 'pre_disk_sync.sh'
POST_BOOTSTRAP_SCRIPT = 'post_bootstrap.sh'
POST_PREPARE_SCRIPT = 'config.sh'
POST_PREPARE_OVERLAY_SCRIPT = 'config-overlay.sh'
POST_HOST_PREPARE_OVERLAY_SCRIPT = 'config-host-overlay.sh'
PRE_CREATE_SCRIPT = 'images.sh'
EDIT_BOOT_CONFIG_SCRIPT = 'edit_boot_config.sh'
EDIT_BOOT_INSTALL_SCRIPT = 'edit_boot_install.sh'
IMAGE_METADATA_DIR = 'image'
ROOT_VOLUME_NAME = 'LVRoot'
SHARED_CACHE_DIR = '/var/cache/kiwi'
MODULE_SPEC: Optional[ModuleSpec] = find_spec('kiwi')
RUNTIME_CHECKER_METADATA = '{}/runtime_checker_metadata.yml'.format(
os.path.dirname(MODULE_SPEC.origin or 'unknown')
) if MODULE_SPEC else 'unknown'
TEMP_DIR = '/var/tmp'
CUSTOM_RUNTIME_CONFIG_FILE = None
PLATFORM_MACHINE = platform.machine()
EFI_FAT_IMAGE_SIZE = 20
log = logging.getLogger('kiwi')
[docs]
class Defaults:
"""
**Implements default values**
Provides static methods for default values and state information
"""
def __init__(self):
self.defaults = {
# alignment in bytes
'kiwi_align': 1048576,
# start sector number
'kiwi_startsector': 2048,
# sectorsize in bytes
'kiwi_sectorsize': 512,
# inode size in bytes for inode based filesystems
'kiwi_inode_size': 256,
# inode ratio for inode based filesystems
'kiwi_inode_ratio': 16384,
# minimum inode number for inode based filesystems
'kiwi_min_inodes': 20000,
# kiwi git revision
'kiwi_revision': __githash__
}
self.profile_key_list = [
'kiwi_align',
'kiwi_startsector',
'kiwi_sectorsize',
'kiwi_revision'
]
[docs]
@staticmethod
def get_luks_key_length():
"""
Provides key length to use for random luks keys
"""
return 256
[docs]
@staticmethod
def get_swapsize_mbytes():
"""
Provides swapsize in MB
"""
return 128
[docs]
@staticmethod
def get_xz_compression_options():
"""
Provides compression options for the xz compressor
:return:
Contains list of options
.. code:: python
['--option=value']
:rtype: list
"""
return [
'--threads=0'
]
[docs]
@staticmethod
def is_x86_arch(arch):
"""
Checks if machine architecture is x86 based
Any arch that matches 32bit and 64bit x86 architecture
causes the method to return True. Anything else will
cause the method to return False
:rtype: bool
"""
x86_arch_names = [
'x86_64', 'i686', 'i586', 'ix86'
]
if arch in x86_arch_names:
return True
return False
[docs]
@staticmethod
def is_ppc64_arch(arch):
"""
Checks if machine architecture is ppc64 based
Any arch that matches little endian or big endian ppc64 architecture
causes the method to return True. Anything else will
cause the method to return False
:rtype: bool
"""
ppc64_arch_names = [
'ppc64', 'ppc64le'
]
if arch in ppc64_arch_names:
return True
return False
[docs]
@staticmethod
def is_buildservice_worker():
"""
Checks if build host is an open buildservice machine
The presence of /.buildenv on the build host indicates
we are building inside of the open buildservice
:return: True if obs worker, else False
:rtype: bool
"""
return os.path.exists(
os.sep + Defaults.get_buildservice_env_name()
)
[docs]
@staticmethod
def get_buildservice_env_name():
"""
Provides the base name of the environment file in a
buildservice worker
:return: file basename
:rtype: str
"""
return '.buildenv'
[docs]
@staticmethod
def get_obs_download_server_url():
"""
Provides the default download server url hosting the public open
buildservice repositories
:return: url path
:rtype: str
"""
return 'http://download.opensuse.org/repositories'
[docs]
@staticmethod
def get_obs_api_server_url():
"""
Provides the default API server url to access the
public open buildservice API
:return: url path
:rtype: str
"""
return 'https://api.opensuse.org'
[docs]
@staticmethod
def get_solvable_location():
"""
Provides the directory to store SAT solvables for repositories.
The solvable files are used to perform package
dependency and metadata resolution
:return: directory path
:rtype: str
"""
return '/var/tmp/kiwi/satsolver'
[docs]
@staticmethod
def set_shared_cache_location(location):
"""
Sets the shared cache location once
:param str location: a location path
"""
global SHARED_CACHE_DIR
SHARED_CACHE_DIR = location
[docs]
@staticmethod
def set_custom_runtime_config_file(filename):
"""
Sets the runtime config file once
:param str filename: a file path name
"""
global CUSTOM_RUNTIME_CONFIG_FILE
CUSTOM_RUNTIME_CONFIG_FILE = filename
[docs]
@staticmethod
def set_temp_location(location):
"""
Sets the temp directory location once
:param str location: a location path
"""
global TEMP_DIR
TEMP_DIR = location
[docs]
@staticmethod
def get_shared_cache_location():
"""
Provides the shared cache location
This is a directory which shares data from the image buildsystem
host with the image root system. The location is returned as an
absolute path stripped off by the leading '/'. This is because
the path is transparently used on the host /<cache-dir> and
inside of the image imageroot/<cache-dir>
:return: directory path
:rtype: str
"""
return os.path.abspath(os.path.normpath(
SHARED_CACHE_DIR
)).lstrip(os.sep)
[docs]
@staticmethod
def get_temp_location():
"""
Provides the base temp directory location
This is the directory used to store any temporary files
and directories created by kiwi during runtime
:return: directory path
:rtype: str
"""
return os.path.abspath(
os.path.normpath(TEMP_DIR)
)
[docs]
@staticmethod
def get_sync_options():
"""
Provides list of default data sync options
:return: list of rsync options
:rtype: list
"""
return [
'--archive', '--hard-links', '--xattrs', '--acls',
'--one-file-system', '--inplace'
]
[docs]
@staticmethod
def get_removed_files_name():
"""
Provides base file name to store removed files
in a delta root build
"""
return 'removed'
[docs]
@staticmethod
def get_exclude_list_for_removed_files_detection() -> List[str]:
"""
Provides list of files/dirs to exclude from the removed
files detection in a delta root build
"""
return [
'etc/hosts.kiwi',
'etc/hosts.sha',
'etc/resolv.conf.kiwi',
'etc/resolv.conf.sha',
'etc/sysconfig/proxy.kiwi',
'etc/sysconfig/proxy.sha',
'usr/lib/sysimage/rpm'
]
[docs]
@staticmethod
def get_exclude_list_for_root_data_sync(no_tmpdirs: bool = True):
"""
Provides the list of files or folders that are created
by KIWI for its own purposes. Those files should be not
be included in the resulting image.
:return: list of file and directory names
:rtype: list
"""
exclude_list = [
'image', '.kconfig'
]
if no_tmpdirs:
exclude_list += ['run/*', 'tmp/*']
exclude_list += [
Defaults.get_buildservice_env_name(),
Defaults.get_shared_cache_location()
]
return exclude_list
[docs]
@staticmethod
def get_exclude_list_from_custom_exclude_files(root_dir: str) -> List:
"""
Provides the list of folders that are excluded by the
optional metadata file image/exclude_files.yaml
:return: list of file and directory names
:param string root_dir: image root directory
:rtype: list
"""
exclude_file = os.sep.join(
[root_dir, 'image', 'exclude_files.yaml']
)
exclude_list = []
if os.path.isfile(exclude_file):
with open(exclude_file) as exclude:
exclude_dict = yaml.safe_load(exclude)
exclude_data = exclude_dict.get('exclude')
if exclude_data and isinstance(exclude_data, list):
for exclude_file in exclude_data:
exclude_list.append(
exclude_file.lstrip(os.sep)
)
else:
log.warning(
f'invalid yaml structure in {exclude_file}, ignored'
)
return exclude_list
[docs]
@staticmethod
def get_exclude_list_for_non_physical_devices():
"""
Provides the list of folders that are not associated
with a physical device. KIWI returns the basename of
the folders typically used as mountpoint for those
devices.
:return: list of file and directory names
:rtype: list
"""
exclude_list = [
'proc', 'sys', 'dev'
]
return exclude_list
[docs]
@staticmethod
def get_failsafe_kernel_options():
"""
Provides failsafe boot kernel options
:return:
list of kernel options
.. code:: python
['option=value', 'option']
:rtype: list
"""
return ' '.join(
[
'ide=nodma',
'apm=off',
'noresume',
'edd=off',
'nomodeset',
'3'
]
)
[docs]
@staticmethod
def get_video_mode_map():
"""
Provides video mode map
Assign a tuple to each kernel vesa hex id for each of the
supported bootloaders
:return:
video type map
.. code:: python
{'kernel_hex_mode': video_type(grub2='mode')}
:rtype: dict
"""
video_type = namedtuple(
'video_type', ['grub2']
)
return {
'0x301': video_type(grub2='640x480'),
'0x310': video_type(grub2='640x480'),
'0x311': video_type(grub2='640x480'),
'0x312': video_type(grub2='640x480'),
'0x303': video_type(grub2='800x600'),
'0x313': video_type(grub2='800x600'),
'0x314': video_type(grub2='800x600'),
'0x315': video_type(grub2='800x600'),
'0x305': video_type(grub2='1024x768'),
'0x316': video_type(grub2='1024x768'),
'0x317': video_type(grub2='1024x768'),
'0x318': video_type(grub2='1024x768'),
'0x307': video_type(grub2='1280x1024'),
'0x319': video_type(grub2='1280x1024'),
'0x31a': video_type(grub2='1280x1024'),
'0x31b': video_type(grub2='1280x1024'),
'auto': video_type(grub2='auto')
}
[docs]
@staticmethod
def get_volume_id():
"""
Provides default value for ISO volume ID
:return: name
:rtype: str
"""
return 'CDROM'
[docs]
@staticmethod
def get_install_volume_id():
"""
Provides default value for ISO volume ID for install media
:return: name
:rtype: str
"""
return 'INSTALL'
[docs]
@staticmethod
def get_snapper_config_template_file(root: str) -> str:
"""
Provides the default configuration template file for snapper.
The location in etc/ are preferred over files in usr/
:return: file path
:rtype: str
"""
snapper_templates = [
'etc/snapper/config-templates/default',
'usr/share/snapper/config-templates/default'
]
snapper_default_conf = ''
for snapper_template in snapper_templates:
template_config = os.path.join(root, snapper_template)
if os.path.exists(template_config):
snapper_default_conf = template_config
break
return snapper_default_conf
[docs]
@staticmethod
def get_default_video_mode():
"""
Uses auto mode for default video. See get_video_mode_map
for details on the value depending which bootloader is
used
:return: auto
:rtype: str
"""
return 'auto'
[docs]
@staticmethod
def get_default_bootloader():
"""
Return default bootloader name which is grub2
:return: bootloader name
:rtype: str
"""
return 'grub2'
[docs]
@staticmethod
def get_grub_custom_arguments(root_dir: str) -> Dict[str, str]:
return {
'grub_directory_name':
Defaults.get_grub_boot_directory_name(root_dir),
'grub_load_command':
'configfile'
}
[docs]
@staticmethod
def get_grub_boot_directory_name(lookup_path):
"""
Provides grub2 data directory name in boot/ directory
Depending on the distribution the grub2 boot path could be
either boot/grub2 or boot/grub. The method will decide for
the correct base directory name according to the name pattern
of the installed grub2 tools
:return: directory basename
:rtype: str
"""
if Path.which(filename='grub2-install', root_dir=lookup_path):
# the presence of grub2-install is an indicator to put all
# grub2 data below boot/grub2
return 'grub2'
else:
# in any other case the assumption is made that all grub
# boot data should live below boot/grub
return 'grub'
[docs]
@staticmethod
def get_grub_basic_modules(multiboot):
"""
Provides list of basic grub modules
:param bool multiboot: grub multiboot mode
:return: list of module names
:rtype: list
"""
modules = [
'ext2',
'iso9660',
'linux',
'echo',
'configfile',
'search_label',
'search_fs_file',
'search',
'search_fs_uuid',
'ls',
'normal',
'gzio',
'png',
'fat',
'gettext',
'font',
'minicmd',
'gfxterm',
'gfxmenu',
'all_video',
'xfs',
'btrfs',
'squash4',
'lvm',
'luks',
'gcry_rijndael',
'gcry_sha256',
'gcry_sha512',
'crypto',
'cryptodisk',
'test',
'true',
'loadenv'
]
if multiboot:
modules.append('multiboot')
return modules
[docs]
@staticmethod
def get_grub_efi_modules(multiboot=False):
"""
Provides list of grub efi modules
:param bool multiboot: grub multiboot mode
:return: list of module names
:rtype: list
"""
host_architecture = Defaults.get_platform_name()
modules = Defaults.get_grub_basic_modules(multiboot) + [
'part_gpt',
'part_msdos',
'efi_gop'
]
if host_architecture == 'x86_64':
modules += [
'efi_uga'
]
return modules
[docs]
@staticmethod
def get_grub_bios_modules(multiboot=False):
"""
Provides list of grub bios modules
:param bool multiboot: grub multiboot mode
:return: list of module names
:rtype: list
"""
modules = Defaults.get_grub_basic_modules(multiboot) + [
'part_gpt',
'part_msdos',
'biosdisk',
'vga',
'vbe',
'chain',
'boot'
]
return modules
[docs]
@staticmethod
def get_grub_ofw_modules():
"""
Provides list of grub ofw modules (ppc)
:return: list of module names
:rtype: list
"""
modules = Defaults.get_grub_basic_modules(multiboot=False) + [
'part_gpt',
'part_msdos',
'boot'
]
return modules
[docs]
@staticmethod
def get_grub_s390_modules():
"""
Provides list of grub ofw modules (s390)
:return: list of module names
:rtype: list
"""
modules = Defaults.get_grub_basic_modules(multiboot=False) + [
'part_gpt',
'part_msdos',
'boot'
]
return modules
[docs]
@staticmethod
def get_grub_path(root_path, filename, raise_on_error=True):
"""
Provides grub path to given search file
Depending on the distribution grub could be installed below
a grub2 or grub directory. grub could also reside in /usr/lib
as well as in /usr/share. Therefore this information needs
to be dynamically looked up
:param string root_path: root path to start the lookup from
:param string filename: filename to search
:param bool raise_on_error: raise on not found, defaults to True
The method returns the path to the given grub search file.
By default it raises a KiwiBootLoaderGrubDataError exception
if the file could not be found in any of the search locations.
If raise_on_error is set to False and no file could be found
the function returns None
:return: filepath
:rtype: str
"""
log.debug(f'Searching grub file: {filename}')
install_dirs = [
'usr/share', 'usr/lib'
]
lookup_list = []
for grub_name in ['grub2', 'grub']:
for install_dir in install_dirs:
grub_path = os.path.join(
root_path, install_dir, grub_name, filename
)
log.debug(f'--> {grub_path}')
if os.path.exists(grub_path):
log.debug(f'--> Found in: {grub_path}')
return grub_path
lookup_list.append(grub_path)
if raise_on_error:
raise KiwiBootLoaderGrubDataError(
'grub path {0} not found in {1}'.format(filename, lookup_list)
)
[docs]
@staticmethod
def get_preparer():
"""
Provides ISO preparer name
:return: name
:rtype: str
"""
return 'KIWI - https://github.com/OSInside/kiwi'
[docs]
@staticmethod
def get_publisher():
"""
Provides ISO publisher name
:return: name
:rtype: str
"""
return 'SUSE LINUX GmbH'
[docs]
@staticmethod
def get_shim_loader(root_path: str) -> Optional[shim_loader_type]:
"""
Provides shim loader file path
Searches distribution specific locations to find shim.efi
below the given root path
:param string root_path: image root path
:return: shim_loader_type | None
:rtype: NamedTuple
"""
shim_pattern_type = namedtuple(
'shim_pattern_type', ['pattern', 'binaryname']
)
shim_file_patterns = [
shim_pattern_type('/usr/lib/shim/shim*.efi.signed.latest', 'shimx64.efi'),
shim_pattern_type('/usr/lib/shim/shim*.efi.signed', 'shimx64.efi'),
shim_pattern_type('/usr/lib/grub/*-efi-signed', 'shimx64.efi'),
shim_pattern_type('/usr/share/efi/*/shim.efi', None),
shim_pattern_type('/usr/lib64/efi/shim.efi', None),
shim_pattern_type('/boot/efi/EFI/*/shim[a-z]*.efi', None),
shim_pattern_type('/boot/efi/EFI/*/shim.efi', None),
shim_pattern_type('/usr/lib/shim/shim*.efi', None)
]
for shim_file_pattern in shim_file_patterns:
for shim_file in sorted(glob.iglob(root_path + shim_file_pattern.pattern), key=len):
if not shim_file_pattern.binaryname:
binaryname = os.path.basename(shim_file)
else:
binaryname = shim_file_pattern.binaryname
return shim_loader_type(
shim_file, binaryname
)
return None
[docs]
@staticmethod
def get_mok_manager(root_path: str) -> Optional[str]:
"""
Provides Mok Manager file path
Searches distribution specific locations to find
the Mok Manager EFI binary
:param str root_path: image root path
:return: file path or None
:rtype: str
"""
mok_manager_file_patterns = [
'/usr/share/efi/*/MokManager.efi',
'/usr/lib64/efi/MokManager.efi',
'/boot/efi/EFI/*/mm*.efi',
'/usr/lib/shim/mm*.efi'
]
for mok_manager_file_pattern in mok_manager_file_patterns:
for mm_file in glob.iglob(root_path + mok_manager_file_pattern):
return mm_file
return None
[docs]
@staticmethod
def get_grub_efi_font_directory(root_path):
"""
Provides distribution specific EFI font directory used with grub.
:param string root_path: image root path
:return: file path or None
:rtype: str
"""
font_dir_patterns = [
'/boot/efi/EFI/*/fonts'
]
for font_dir_pattern in font_dir_patterns:
for font_dir in glob.iglob(root_path + font_dir_pattern):
return font_dir
[docs]
@staticmethod
def get_unsigned_grub_loader(
root_path: str, target_type: str = 'disk'
) -> Optional[str]:
"""
Provides unsigned grub efi loader file path
Searches distribution specific locations to find a distro
grub EFI binary within the given root path
:param string root_path: image root path
:return: file path or None
:rtype: str
"""
unsigned_grub_file = None
unsigned_grub_file_patterns = {
'disk': [
'/usr/share/grub*/*-efi/grub.efi',
'/usr/lib/grub*/*-efi/grub.efi',
'/boot/efi/EFI/*/grubx64.efi',
'/boot/efi/EFI/*/grubaa64.efi'
],
'iso': [
'/boot/efi/EFI/*/gcdx64.efi',
'/usr/share/grub*/*-efi/grub.efi',
'/usr/lib/grub*/*-efi/grub.efi',
'/boot/efi/EFI/*/grubx64.efi',
'/boot/efi/EFI/*/grubaa64.efi'
]
}
for unsigned_grub_file_pattern in unsigned_grub_file_patterns[target_type]:
for unsigned_grub_file in glob.iglob(
root_path + unsigned_grub_file_pattern
):
return unsigned_grub_file
return None
[docs]
@staticmethod
def get_grub_bios_core_loader(root_path):
"""
Provides grub bios image
Searches distribution specific locations to find the
core bios image below the given root path
:param string root_path: image root path
:return: file path or None
:rtype: str
"""
bios_grub_core_patterns = [
'/usr/share/grub*/{0}/{1}'.format(
Defaults.get_bios_module_directory_name(), Defaults.get_bios_image_name()
),
'/usr/lib/grub*/{0}/{1}'.format(
Defaults.get_bios_module_directory_name(), Defaults.get_bios_image_name()
)
]
for bios_grub_core_pattern in bios_grub_core_patterns:
for bios_grub_core in glob.iglob(
root_path + bios_grub_core_pattern
):
return bios_grub_core
[docs]
@staticmethod
def get_iso_grub_loader():
"""
Return name of eltorito grub image used as ISO loader
:return: file base name
:rtype: str
"""
return 'eltorito.img'
[docs]
@staticmethod
def get_iso_grub_mbr():
"""
Return name of hybrid MBR image used as ISO boot record
:return: file base name
:rtype: str
"""
return 'boot_hybrid.img'
[docs]
@staticmethod
def get_signed_grub_loader(
root_path: str, target_type: str = 'disk'
) -> Optional[grub_loader_type]:
"""
Provides shim signed grub loader file path
Searches distribution specific locations to find a grub
EFI binary within the given root path
:param str root_path: image root path
:return: grub_loader_type | None
:rtype: NamedTuple
"""
grub_pattern_type = namedtuple(
'grub_pattern_type', ['pattern', 'binaryname']
)
signed_grub_file_patterns = {
'disk': [
grub_pattern_type(
'/usr/share/efi/*/grub.efi', None
),
grub_pattern_type(
'/usr/lib64/efi/grub.efi', None
),
grub_pattern_type(
'/boot/efi/EFI/*/grub*.efi', None
),
grub_pattern_type(
'/usr/share/grub*/*-efi/grub.efi', None
),
grub_pattern_type(
'/usr/lib/grub/x86_64-efi-signed/grubx64.efi.signed',
'grubx64.efi'
)
],
'iso': [
grub_pattern_type(
'/boot/efi/EFI/*/gcdx64.efi',
'grubx64.efi'
),
grub_pattern_type(
'/boot/efi/EFI/*/gcdaa64.efi',
'grubaa64.efi'
),
grub_pattern_type(
'/usr/share/efi/*/grub.efi', None
),
grub_pattern_type(
'/usr/lib64/efi/grub.efi', None
),
grub_pattern_type(
'/boot/efi/EFI/*/grub*.efi', None
),
grub_pattern_type(
'/usr/share/grub*/*-efi/grub.efi', None
),
grub_pattern_type(
'/usr/lib/grub/x86_64-efi-signed/grubx64.efi.signed',
'grubx64.efi'
)
]
}
for signed_grub in signed_grub_file_patterns[target_type]:
for signed_grub_file in glob.iglob(root_path + signed_grub.pattern):
if not signed_grub.binaryname:
binaryname = os.path.basename(signed_grub_file)
else:
binaryname = signed_grub.binaryname
return grub_loader_type(
signed_grub_file, binaryname
)
return None
[docs]
@staticmethod
def get_efi_vendor_directory(efi_path):
"""
Provides EFI vendor directory if present
Looks up distribution specific EFI vendor directory
:param string root_path: path to efi mountpoint
:return: directory path or None
:rtype: str
"""
efi_vendor_directories = [
'EFI/fedora',
'EFI/redhat',
'EFI/centos',
'EFI/opensuse',
'EFI/ubuntu',
'EFI/debian'
]
for efi_vendor_directory in efi_vendor_directories:
efi_vendor_directory = os.sep.join([efi_path, efi_vendor_directory])
if os.path.exists(efi_vendor_directory):
return efi_vendor_directory
[docs]
@staticmethod
def get_vendor_grubenv(efi_path):
efi_vendor_directory = Defaults.get_efi_vendor_directory(efi_path)
if efi_vendor_directory:
grubenv = os.sep.join([efi_vendor_directory, 'grubenv'])
if os.path.exists(grubenv):
return grubenv
[docs]
@staticmethod
def get_shim_vendor_directory(root_path):
"""
Provides shim vendor directory
Searches distribution specific locations to find shim.efi
below the given root path and return the directory name
to the file found
:param string root_path: image root path
:return: directory path or None
:rtype: str
"""
shim_vendor_patterns = [
'/boot/efi/EFI/*/shim*.efi',
'/EFI/*/shim*.efi'
]
for shim_vendor_pattern in shim_vendor_patterns:
for shim_file in glob.iglob(root_path + shim_vendor_pattern):
return os.path.dirname(shim_file)
[docs]
@staticmethod
def get_default_volume_group_name():
"""
Provides default LVM volume group name
:return: name
:rtype: str
"""
return 'systemVG'
[docs]
@staticmethod
def get_min_partition_mbytes():
"""
Provides default minimum partition size in mbytes
:return: mbsize value
:rtype: int
"""
return 10
[docs]
@staticmethod
def get_min_volume_mbytes(filesystem: str):
"""
Provides default minimum LVM volume size in mbytes
per filesystem
:return: mbsize value
:rtype: int
"""
if filesystem == 'btrfs':
return 120
elif filesystem == 'xfs':
return 300
else:
return 30
[docs]
@staticmethod
def get_lvm_overhead_mbytes():
"""
Provides empiric LVM overhead size in mbytes
:return: mbsize value
:rtype: int
"""
return 80
[docs]
@staticmethod
def get_default_boot_mbytes():
"""
Provides default boot partition size in mbytes
:return: mbsize value
:rtype: int
"""
return 300
[docs]
@staticmethod
def get_default_efi_boot_mbytes():
"""
Provides default EFI partition size in mbytes
:return: mbsize value
:rtype: int
"""
return 20
[docs]
@staticmethod
def get_recovery_spare_mbytes():
"""
Provides spare size of recovery partition in mbytes
:return: mbsize value
:rtype: int
"""
return 300
[docs]
@staticmethod
def get_default_legacy_bios_mbytes():
"""
Provides default size of bios_grub partition in mbytes
:return: mbsize value
:rtype: int
"""
return 2
[docs]
@staticmethod
def get_default_prep_mbytes():
"""
Provides default size of prep partition in mbytes
:return: mbsize value
:rtype: int
"""
return 8
[docs]
@staticmethod
def get_vagrant_config_virtualbox_guest_additions():
"""
Provides the default value for
``vagrantconfig.virtualbox_guest_additions_present``
:return: whether guest additions are expected to be present in the
vagrant box
:rtype: bool
"""
return False
[docs]
@staticmethod
def get_firmware_types():
"""
Provides supported architecture specific firmware types
:return: firmware types per architecture
:rtype: dict
"""
return {
'x86_64': ['efi', 'uefi', 'bios', 'ec2'],
'i586': ['bios'],
'i686': ['bios'],
'ix86': ['bios'],
'aarch64': ['efi', 'uefi'],
'arm64': ['efi', 'uefi'],
'armv5el': ['efi', 'uefi'],
'armv5tel': ['efi', 'uefi'],
'armv6hl': ['efi', 'uefi'],
'armv6l': ['efi', 'uefi'],
'armv7hl': ['efi', 'uefi'],
'armv7l': ['efi', 'uefi'],
'armv8l': ['efi', 'uefi'],
'ppc': ['ofw'],
'ppc64': ['ofw', 'opal'],
'ppc64le': ['ofw', 'opal'],
'riscv64': ['efi', 'uefi'],
's390': [],
's390x': []
}
[docs]
@staticmethod
def get_default_firmware(arch):
"""
Provides default firmware for specified architecture
:param string arch: machine architecture name
:return: firmware name
:rtype: str
"""
default_firmware = {
'x86_64': 'bios',
'i586': 'bios',
'i686': 'bios',
'ix86': 'bios',
'ppc': 'ofw',
'ppc64': 'ofw',
'ppc64le': 'ofw',
'arm64': 'efi',
'aarch64': 'efi',
'armv5el': 'efi',
'armv5tel': 'efi',
'armv6hl': 'efi',
'armv6l': 'efi',
'armv7hl': 'efi',
'armv7l': 'efi',
'armv8l': 'efi',
'riscv64': 'efi'
}
if arch in default_firmware:
return default_firmware[arch]
[docs]
@staticmethod
def get_efi_capable_firmware_names():
"""
Provides list of EFI capable firmware names. These are
those for which kiwi supports the creation of an EFI
bootable disk image
:return: firmware names
:rtype: list
"""
return ['efi', 'uefi']
[docs]
@staticmethod
def get_ec2_capable_firmware_names():
"""
Provides list of EC2 capable firmware names. These are
those for which kiwi supports the creation of disk images
bootable within the Amazon EC2 public cloud
:return: firmware names
:rtype: list
"""
return ['ec2']
[docs]
@staticmethod
def get_efi_module_directory_name(arch):
"""
Provides architecture specific EFI directory name which
stores the EFI binaries for the desired architecture.
:param string arch: machine architecture name
:return: directory name
:rtype: str
"""
default_module_directory_names = {
'x86_64': 'x86_64-efi',
# There is no dedicated xen architecture but there are
# modules provided for xen. Thus we treat it as an
# architecture
'x86_64_xen': 'x86_64-xen',
'aarch64': 'arm64-efi',
'arm64': 'arm64-efi',
'armv5el': 'arm-efi',
'armv5tel': 'arm-efi',
'armv6l': 'arm-efi',
'armv7l': 'arm-efi',
'armv8l': 'arm-efi',
'riscv64': 'riscv64-efi'
}
if arch in default_module_directory_names:
return default_module_directory_names[arch]
[docs]
@staticmethod
def get_bios_module_directory_name():
"""
Provides BIOS directory name which stores the pc binaries
:return: directory name
:rtype: str
"""
return 'powerpc-ieee1275' if Defaults.is_ppc64_arch(Defaults.get_platform_name()) else 'i386-pc'
[docs]
@staticmethod
def get_efi_image_name(arch):
"""
Provides architecture specific EFI boot binary name
:param string arch: machine architecture name
:return: name
:rtype: str
"""
default_efi_image_names = {
'x86_64': 'bootx64.efi',
'aarch64': 'bootaa64.efi',
'arm64': 'bootaa64.efi',
'armv5el': 'bootarm.efi',
'armv5tel': 'bootarm.efi',
'armv6l': 'bootarm.efi',
'armv7l': 'bootarm.efi',
'armv8l': 'bootarm.efi',
'riscv64': 'bootriscv64.efi'
}
if arch in default_efi_image_names:
return default_efi_image_names[arch]
[docs]
@staticmethod
def get_bios_image_name():
"""
Provides bios core boot binary name
:return: name
:rtype: str
"""
return 'grub.elf' if Defaults.is_ppc64_arch(Defaults.get_platform_name()) else 'core.img'
[docs]
@staticmethod
def get_default_boot_timeout_seconds():
"""
Provides default boot timeout in seconds
:return: seconds
:rtype: int
"""
return 10
[docs]
@staticmethod
def get_default_disk_start_sector():
"""
Provides the default initial disk sector for the first disk
partition.
:return: sector value
:rtype: int
"""
return Defaults().defaults['kiwi_startsector']
[docs]
@staticmethod
def get_default_efi_partition_table_type():
"""
Provides the default partition table type for efi firmwares.
:return: partition table type name
:rtype: str
"""
return 'gpt'
[docs]
@staticmethod
def get_default_inode_size():
"""
Provides default size of inodes in bytes. This is only
relevant for inode based filesystems
:return: bytesize value
:rtype: int
"""
return Defaults().defaults['kiwi_inode_size']
[docs]
@staticmethod
def get_archive_image_types():
"""
Provides list of supported archive image types
:return: archive names
:rtype: list
"""
return ['tbz', 'cpio']
[docs]
@staticmethod
def get_container_image_types():
"""
Provides list of supported container image types
:return: container names
:rtype: list
"""
return ['docker', 'oci', 'appx']
[docs]
@staticmethod
def get_filesystem_image_types():
"""
Provides list of supported filesystem image types
:return: filesystem names
:rtype: list
"""
return [
'ext2', 'ext3', 'ext4', 'btrfs', 'squashfs',
'xfs', 'fat16', 'fat32', 'erofs'
]
[docs]
@staticmethod
def get_default_live_iso_type():
"""
Provides default live iso union type
:return: live iso type
:rtype: str
"""
return 'overlay'
[docs]
@staticmethod
def get_default_uri_type():
"""
Provides default URI type
Absolute path specifications used in the context of an URI
will apply the specified default mime type
:return: URI mime type
:rtype: str
"""
return 'dir:/'
[docs]
@staticmethod
def get_dracut_conf_name():
"""
Provides file path of dracut config file to be used with KIWI
:return: file path name
:rtype: str
"""
return '/etc/dracut.conf.d/02-kiwi.conf'
[docs]
@staticmethod
def get_live_dracut_modules_from_flag(flag_name):
"""
Provides flag_name to dracut modules name map
Depending on the value of the flag attribute in the KIWI image
description specific dracut modules need to be selected
:return: dracut module names as list
:rtype: list
"""
live_modules = {
'overlay': ['kiwi-live'],
'dmsquash': ['dmsquash-live', 'livenet']
}
if flag_name in live_modules:
return live_modules[flag_name]
else:
return ['kiwi-live']
[docs]
@staticmethod
def get_default_live_iso_root_filesystem():
"""
Provides default live iso root filesystem type
:return: filesystem name
:rtype: str
"""
return 'ext4'
[docs]
@staticmethod
def get_live_iso_persistent_boot_options(persistent_filesystem=None):
"""
Provides list of boot options passed to the dracut
kiwi-live module to setup persistent writing
:return: list of boot options
:rtype: list
"""
live_iso_persistent_boot_options = [
'rd.live.overlay.persistent'
]
if persistent_filesystem:
live_iso_persistent_boot_options.append(
'rd.live.overlay.cowfs={0}'.format(persistent_filesystem)
)
return live_iso_persistent_boot_options
[docs]
@staticmethod
def get_disk_image_types():
"""
Provides supported disk image types
:return: disk image type names
:rtype: list
"""
return ['oem']
[docs]
@staticmethod
def get_live_image_types():
"""
Provides supported live image types
:return: live image type names
:rtype: list
"""
return ['iso']
[docs]
@staticmethod
def get_kis_image_types():
"""
Provides supported kis image types
:return: kis image type names
:rtype: list
"""
return ['kis', 'pxe']
[docs]
@staticmethod
def get_enclaves_image_types():
"""
Provides supported enclave(initrd-only) image types
:return: enclave image type names
:rtype: list
"""
return ['enclave']
[docs]
@staticmethod
def get_boot_image_description_path():
"""
Provides the path to find custom kiwi boot descriptions
:return: directory path
:rtype: str
"""
return '/usr/share/kiwi/custom_boot'
[docs]
@staticmethod
def get_boot_image_strip_file():
"""
Provides the file path to bootloader strip metadata.
This file contains information about the files and directories
automatically striped out from the kiwi initrd
:return: file path
:rtype: str
"""
return Defaults.project_file('config/strip.xml')
[docs]
@staticmethod
def get_schema_file():
"""
Provides file path to kiwi RelaxNG schema
:return: file path
:rtype: str
"""
return Defaults.project_file('schema/kiwi.rng')
[docs]
@staticmethod
def get_common_functions_file():
"""
Provides the file path to config functions metadata.
This file contains bash functions used for system
configuration or in the boot code from the kiwi initrd
:return: file path
:rtype: str
"""
return Defaults.project_file('config/functions.sh')
[docs]
@staticmethod
def get_xsl_stylesheet_file():
"""
Provides the file path to the KIWI XSLT style sheets
:return: file path
:rtype: str
"""
return Defaults.project_file('xsl/master.xsl')
[docs]
@staticmethod
def get_schematron_module_name():
"""
Provides module name for XML SchemaTron validations
:return: python module name
:rtype: str
"""
return 'lxml.isoschematron'
[docs]
@staticmethod
def project_file(filename):
"""
Provides the python module base directory search path
The method uses the importlib.resources.path method to identify
files and directories from the application
:param string filename: relative project file
:return: absolute file path name
:rtype: str
"""
with as_file(importlib.resources.files('kiwi')) as path:
return f'{path}/{filename}'
[docs]
@staticmethod
def get_imported_root_image(root_dir):
"""
Provides the path to an imported root system image
If the image description specified a derived_from attribute
the file from this attribute is copied into the root_dir
using the name as provided by this method
:param string root_dir: image root directory
:return: file path name
:rtype: str
"""
return os.sep.join([root_dir, 'image', 'imported_root'])
[docs]
@staticmethod
def get_iso_boot_path():
"""
Provides arch specific relative path to boot files
on kiwi iso filesystems
:return: relative path name
:rtype: str
"""
return os.sep.join(
['boot', Defaults.get_platform_name()]
)
[docs]
@staticmethod
def get_container_compression():
"""
Provides default container compression
:return: True
:rtype: bool
"""
return True
[docs]
@staticmethod
def get_default_container_name():
"""
Provides the default container name.
:return: name
:rtype: str
"""
return 'kiwi-container'
[docs]
@staticmethod
def get_container_base_image_tag():
"""
Provides the tag used to identify base layers during the build
of derived images.
:return: tag
:rtype: str
"""
return 'base_layer'
[docs]
@staticmethod
def get_default_container_tag():
"""
Provides the default container tag.
:return: tag
:rtype: str
"""
return 'latest'
[docs]
@staticmethod
def get_default_container_subcommand():
"""
Provides the default container subcommand.
:return: command as a list of arguments
:rtype: list
"""
return ['/bin/bash']
[docs]
@staticmethod
def get_default_container_created_by():
"""
Provides the default 'created by' history entry for containers.
:return: the specific kiwi version used for the build
:rtype: str
"""
return 'KIWI {0}'.format(__version__)
[docs]
@staticmethod
def get_custom_rpm_macros_path():
"""
Returns the custom macros directory for the rpm database.
:return: path name
:rtype: str
"""
return 'usr/lib/rpm/macros.d'
[docs]
@staticmethod
def get_custom_rpm_bootstrap_macro_name():
"""
Returns the rpm bootstrap macro file name created
in the custom rpm macros path
:return: filename
:rtype: str
"""
return 'macros.kiwi-bootstrap-config'
[docs]
@staticmethod
def get_custom_rpm_image_macro_name():
"""
Returns the rpm image macro file name created
in the custom rpm macros path
:return: filename
:rtype: str
"""
return 'macros.kiwi-image-config'
[docs]
@staticmethod
def get_default_package_manager() -> str:
"""
Returns the default package manager name if none
is configured in the image description
:return: package manager name
:rtype: str
"""
return 'dnf4'
[docs]
@staticmethod
def get_discoverable_partition_ids() -> Dict[str, str]:
"""
Provides arch specific partition UUIDs as defined
by the UAPI group
:return: partition UUIDs
:rtype: dict
"""
arch = Defaults.get_platform_name()
part_uuids_archs = {
'x86_64': {
'root':
'4f68bce3e8cd4db196e7fbcaf984b709',
'usr':
'8484680c952148c69c11b0720656f69e',
'usr-verity':
'77ff5f63e7b64633acf41565b864c0e6'
},
'ix86': {
'root':
'44479540f29741b29af7d131d5f0458a',
'usr':
'75250d768cc6458ebd66bd47cc81a812',
'usr-verity':
'8f461b0d14ee4e819aa9049b6fb97abd'
},
'aarch64': {
'root':
'b921b0451df041c3af444c6f280d3fae',
'usr':
'b0e01050ee5f4390949a9101b17104e9',
'usr-verity':
'6e11a4e7fbca4dedb9e9e1a512bb664e'
},
'riscv64': {
'root':
'72ec70a6cf7440e6bd494bda08e8f224',
'usr':
'beaec34b8442439ba40b984381ed097d',
'usr-verity':
'8f1056be9b0547c481d6be53128e5b54'
}
}
part_uuids_arch = part_uuids_archs.get(arch) or {}
return {
'root':
part_uuids_arch.get('root') or '',
'usr':
part_uuids_arch.get('usr') or '',
'usr-verity':
part_uuids_arch.get('usr-verity') or '',
'usr-secondary':
'75250d768cc6458ebd66bd47cc81a812',
'usr-secondary-verity':
'8f461b0d14ee4e819aa9049b6fb97abd',
'esp':
'c12a7328f81f11d2ba4b00a0c93ec93b',
'xbootldr':
'bc13c2ff59e64262a352b275fd6f7172',
'swap':
'0657fd6da4ab43c484e50933c84b4f4f',
'home':
'933ac7e12eb44f13b8440e14e2aef915',
'srv':
'3b8f842520e04f3b907f1a25a76f98e8',
'var':
'4d21b016b53445c2a9fb5c16e091fd2d',
'tmp':
'7ec6f5573bc54acab29316ef5df639d1',
'user-home':
'773f91ef66d449b5bd83d683bf40ad16',
'linux-generic':
'0fc63daf848347728e793d69d8477de4'
}
[docs]
@staticmethod
def get_bls_loader_entries_dir() -> str:
"""
Provide default loader entries directory for BLS loaders
:return: directory name
:rtype: str
"""
return '/boot/loader/entries'
[docs]
def get(self, key):
"""
Implements get method for profile elements
:param string key: profile keyname
:return: key value
:rtype: str
"""
if key in self.defaults:
return self.defaults[key]
[docs]
@staticmethod
def get_profile_file(root_dir):
"""
Return name of profile file for given root directory
"""
return root_dir + '/.profile'
[docs]
def to_profile(self, profile):
"""
Implements method to add list of profile keys and their values
to the specified instance of a Profile class
:param object profile: Profile instance
"""
for key in sorted(self.profile_key_list):
# Do not apply default values to any variable that was
# already defined in the profile instance.
cur_profile = profile.dot_profile
if key not in cur_profile or cur_profile[key] is None:
profile.add(key, self.get(key))