Source code for kiwi.boot.image.base

# 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 re
import os
import logging
import glob
from typing import (
    Optional, List, Dict, NamedTuple
)

# project
from kiwi.defaults import Defaults
from kiwi.system.setup import SystemSetup
from kiwi.system.identifier import SystemIdentifier
from kiwi.xml_description import XMLDescription
from kiwi.xml_state import XMLState
from kiwi.path import Path
from kiwi.system.kernel import Kernel

from kiwi.exceptions import (
    KiwiTargetDirectoryNotFound,
    KiwiConfigFileNotFound,
    KiwiDiskBootImageError
)

boot_names_type = NamedTuple(
    'boot_names_type', [
        ('kernel_name', str),
        ('initrd_name', str),
        ('kernel_version', str),
        ('kernel_filename', str)
    ]
)

log = logging.getLogger('kiwi')


[docs] class BootImageBase: """ **Base class for boot image(initrd) task** """ def __init__( self, xml_state: XMLState, target_dir: str, root_dir: Optional[str] = None, signing_keys: Optional[List[str]] = None, target_arch: Optional[str] = None ) -> None: """ Create a base class instance of a BootImage The base class instance only support the common operations for boot images. The actual preparation and creation of an initrd is specific to the used initrd creation toolchain :param XMLState xml_state: Instance of :class:`XMLState` :param str target_dir: target dir to store the initrd :param str root_dir: system image root directory :param List[str] signing_keys: list of package signing keys :param str target_arch: target architecture """ self.xml_state = xml_state self.target_dir = target_dir self.initrd_filename = '' self.boot_xml_state: Optional[XMLState] = None self.setup: Optional[SystemSetup] = None self.signing_keys = signing_keys self.target_arch = target_arch self.boot_root_directory = root_dir or '' if not os.path.exists(target_dir): raise KiwiTargetDirectoryNotFound( 'target directory %s not found' % target_dir ) self.initrd_base_name = ''.join( [ self.xml_state.xml_data.get_name(), '.' + Defaults.get_platform_name(), '-' + self.xml_state.get_image_version(), '.initrd' ] ) self.post_init()
[docs] def post_init(self) -> None: """ Post initialization method Implementation in specialized boot image class """ pass
[docs] @staticmethod def has_initrd_support() -> bool: """ Indicates if this instance supports actual creation of an initrd The method needs to be overwritten by the subclass implementing preparation and creation of an initrd """ return False
[docs] def include_file(self, filename: str, install_media: bool = False) -> None: """ Include file to boot image For kiwi boot images this is done by adding package or archive definitions with the bootinclude attribute. Thus for kiwi boot images the method is a noop :param str filename: file path name :param bool install_media: include also for installation media initrd """ pass
[docs] def include_module(self, module: str, install_media: bool = False) -> None: """ Include module to boot image For kiwi boot no modules configuration is required. Thus in such a case this method is a noop. :param str module: module to include :param bool install_media: include the module for install initrds """ pass
[docs] def omit_module(self, module: str, install_media: bool = False) -> None: """ Omit module to boot image For kiwi boot no modules configuration is required. Thus in such a case this method is a noop. :param str module: module to omit :param bool install_media: omit the module for install initrds """ pass
[docs] def set_static_modules( self, modules: List[str], install_media: bool = False ) -> None: """ Set static modules list for boot image For kiwi boot no modules configuration is required. Thus in such a case this method is a noop. :param list modules: list of modules to include :param bool install_media: lists the modules for install initrds """ pass
[docs] def write_system_config_file( self, config: Dict, config_file: Optional[str] = None ) -> None: """ Writes relevant boot image configuration into configuration file that will be part of the system image. This is used to configure any further boot image rebuilds after deployment. For instance, initrds recreated on kernel update. For kiwi boot no specific configuration is required for initrds recreation, thus this method is a noop in that case. :param dict config: dictonary including configuration parameters :param str config_file: configuration file to write """ pass
[docs] def get_boot_names(self) -> boot_names_type: """ Provides kernel and initrd names for the boot image :return: Contains boot_names_type tuple .. code:: python boot_names_type( kernel_name='INSTALLED_KERNEL', initrd_name='DRACUT_OUTPUT_NAME', kernel_version='KERNEL_VERSION', kernel_filename='KERNEL_FILE_NAME' ) :rtype: boot_names_type """ kernel = Kernel( self.boot_root_directory ) kernel_info = kernel.get_kernel() if not kernel_info: if self.xml_state.get_initrd_system() == 'none': return boot_names_type( kernel_name='none', initrd_name='none', kernel_version='none', kernel_filename='none' ) raise KiwiDiskBootImageError( 'No kernel in boot image tree %s found' % self.boot_root_directory ) dracut_output_format = self._get_boot_image_output_file_format( kernel_info.version ) return boot_names_type( kernel_version=kernel_info.version, kernel_name=kernel_info.name, initrd_name=dracut_output_format.format( kernel_version=kernel_info.version ), kernel_filename=kernel_info.filename )
[docs] def prepare(self) -> None: """ Prepare new root system to create initrd from. Implementation is only needed if there is no other root system available Implementation in specialized boot image class """ raise NotImplementedError
[docs] def create_initrd( self, mbrid: Optional[SystemIdentifier] = None, basename: Optional[str] = None, install_initrd: bool = False ) -> None: """ Implements creation of the initrd :param SystemIdentifier mbrid: instance of SystemIdentifier :param str basename: base initrd file name :param bool install_initrd: installation media initrd Implementation in specialized boot image class """ raise NotImplementedError
[docs] def is_prepared(self) -> bool: """ Check if initrd system is prepared. :return: True or False :rtype: bool """ return bool(os.listdir(self.boot_root_directory))
[docs] def load_boot_xml_description(self) -> None: """ Load the boot image description referenced by the system image description boot attribute """ log.info('Loading Boot XML description') boot_description_directory = self.get_boot_description_directory() if not boot_description_directory: raise KiwiConfigFileNotFound( 'no boot reference specified in XML description' ) boot_config_file = boot_description_directory + '/config.xml' if not os.path.exists(boot_config_file): raise KiwiConfigFileNotFound( 'no Boot XML description found in %s' % boot_description_directory ) boot_description = XMLDescription( description=boot_config_file, derived_from=self.xml_state.xml_data.description_dir ) boot_image_profile = self.xml_state.build_type.get_bootprofile() if not boot_image_profile: boot_image_profile = 'default' boot_kernel_profile = self.xml_state.build_type.get_bootkernel() if not boot_kernel_profile: boot_kernel_profile = 'std' self.boot_xml_state = XMLState( boot_description.load(), [boot_image_profile, boot_kernel_profile] ) log.info('--> loaded %s', boot_config_file) if self.boot_xml_state.build_type: log.info( '--> Selected build type: %s', self.boot_xml_state.get_build_type_name() ) if self.boot_xml_state.profiles: log.info( '--> Selected boot profiles: image: %s, kernel: %s', boot_image_profile, boot_kernel_profile )
[docs] def import_system_description_elements(self) -> None: """ Copy information from the system image relevant to create the boot image to the boot image state XML description """ self.xml_state.copy_displayname( self.boot_xml_state ) self.xml_state.copy_name( self.boot_xml_state ) self.xml_state.copy_repository_sections( target_state=self.boot_xml_state, wipe=True ) self.xml_state.copy_drivers_sections( self.boot_xml_state ) strip_description = XMLDescription( Defaults.get_boot_image_strip_file() ) strip_xml_state = XMLState(strip_description.load()) strip_xml_state.copy_strip_sections( self.boot_xml_state ) self.xml_state.copy_strip_sections( self.boot_xml_state ) preferences_subsection_names = [ 'bootloader_theme', 'bootsplash_theme', 'locale', 'packagemanager', 'rpm_check_signatures', 'rpm_excludedocs', 'showlicense' ] self.xml_state.copy_preferences_subsections( preferences_subsection_names, self.boot_xml_state ) self.xml_state.copy_bootincluded_packages( self.boot_xml_state ) self.xml_state.copy_bootincluded_archives( self.boot_xml_state ) self.xml_state.copy_bootdelete_packages( self.boot_xml_state ) type_attributes = [ 'bootkernel', 'bootprofile', 'btrfs_root_is_snapshot', 'gpt_hybrid_mbr', 'devicepersistency', 'filesystem', 'firmware', 'fsmountoptions', 'hybridpersistent', 'hybridpersistent_filesystem', 'initrd_system', 'installboot', 'installprovidefailsafe', 'kernelcmdline', 'ramonly', 'target_removable', 'vga', 'wwid_wait_timeout' ] self.xml_state.copy_build_type_attributes( type_attributes, self.boot_xml_state ) self.xml_state.copy_bootloader_section( self.boot_xml_state ) self.xml_state.copy_systemdisk_section( self.boot_xml_state ) self.xml_state.copy_machine_section( self.boot_xml_state ) self.xml_state.copy_oemconfig_section( self.boot_xml_state )
[docs] def get_boot_description_directory(self) -> Optional[str]: """ Provide path to the boot image XML description :return: path name :rtype: str """ boot_description = self.xml_state.build_type.get_boot() if boot_description: if not boot_description[0] == '/': boot_description = \ Defaults.get_boot_image_description_path() + '/' + \ boot_description return boot_description
[docs] def cleanup(self) -> None: """ Cleanup temporary boot image data if any """ pass
def _get_boot_image_output_file_format(self, kernel_version: str) -> str: """ The initrd output file format varies between the different Linux distributions. Tools like lsinitrd, and also grub2 rely on the initrd output file to be in that format. Thus kiwi should use the same file format to stay compatible with the distributions. The format is determined in three stages: a) check for an existing initrd file and use this naming schema b) if no initrd file is found check the dracut binary for its default output file name schema c) if no output file schema could be detected from the dracut binary return a default output file format which is initramfs-{kernel_version}.img Let's hope all this mess can be deleted once all distros just can agree on one initrd file name schema """ if self.xml_state.get_initrd_system() == 'kiwi': # The custom kiwi initrd system is used only on SUSE systems. # The initrd environment does not provide dracut and thus the # outfile format cannot be determined. on SUSE systems the # initrd format is different than on upstream and therefore # it can be explicitly specified. Note that the custom initrd # system will become obsolete in the future. default_outfile_format = 'initrd-{kernel_version}' else: default_outfile_format = 'initramfs-{kernel_version}.img' outfile_format = \ self._get_boot_image_output_file_format_from_existing_file( kernel_version ) if not outfile_format: outfile_format = \ self._get_boot_image_output_file_format_from_dracut_code() if outfile_format: return outfile_format else: log.warning('Could not detect dracut output file format') log.warning( 'Using default initrd file name format {0}'.format( default_outfile_format ) ) return default_outfile_format def _get_boot_image_output_file_format_from_dracut_code( self ) -> Optional[str]: dracut_tool = Path.which( 'dracut', root_dir=self.boot_root_directory, access_mode=os.X_OK ) if dracut_tool: outfile_expression = r'outfile=".*/boot/(init.*kernel.*)"' with open(dracut_tool) as dracut: matches = re.findall(outfile_expression, dracut.read()) if matches: return matches[0].replace( '$kernel', '{kernel_version}' ).replace( '${kernel}', '{kernel_version}' ) return None def _get_boot_image_output_file_format_from_existing_file( self, kernel_version: str ) -> Optional[str]: for initrd_file in glob.iglob(self.boot_root_directory + '/boot/init*'): if not os.path.islink(initrd_file): return os.path.basename(initrd_file).replace( kernel_version, '{kernel_version}' ) return None