# -*- Mode:Python; indent-tabs-mode:nil; tab-width:4 -*-
#
# Copyright 2021-2024 Canonical Ltd.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License version 3 as published by the Free Software Foundation.
#
# 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
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
"""The parts lifecycle manager."""
import os
import re
import sys
from collections.abc import Sequence
from pathlib import Path
from typing import Any, cast
from pydantic import ValidationError
from craft_parts import errors, executor, packages, plugins, sequencer
from craft_parts.actions import Action
from craft_parts.dirs import ProjectDirs
from craft_parts.features import Features
from craft_parts.infos import ProjectInfo
from craft_parts.overlays import LayerHash
from craft_parts.parts import Part, part_by_name
from craft_parts.state_manager import states
from craft_parts.steps import Step
from craft_parts.utils.partition_utils import validate_partition_names
[docs]
class LifecycleManager:
"""Coordinate the planning and execution of the parts lifecycle.
The lifecycle manager determines the list of actions that needs be executed in
order to obtain a tree of installed files from the specification on how to
process its parts, and provides a mechanism to execute each of these actions.
:param all_parts: A dictionary containing the parts specification according
to the :ref:`parts schema<part_properties>`. The format is compatible with the
output generated by PyYAML's ``yaml.load``.
:param application_name: A unique non-empty identifier for the application
using Craft Parts. Valid application names contain upper and lower case
letters, underscores or numbers, and must start with a letter.
:param project_name: name of the project being built.
:param cache_dir: The path to store cached packages and files. If not
specified, a directory under the application name entry in the XDG base
directory will be used.
:param work_dir: The toplevel directory for work directories. The current
directory will be used if none is specified.
:param arch: The architecture to build for. Defaults to the host system
architecture.
:param base: [deprecated] The system base the project being processed will
run on. Defaults to the system where Craft Parts is being executed.
:param parallel_build_count: The maximum number of concurrent jobs to be
used to build each part of this project.
:param application_package_name: The name of the application package, if required
by the package manager used by the platform. Defaults to the application name.
:param ignore_local_sources: A list of local source patterns to ignore.
:param extra_build_packages: A list of additional build packages to install.
:param extra_build_snaps: A list of additional build snaps to install.
:param track_stage_packages: Add primed stage packages to the prime state.
:param strict_mode: Only allow plugins capable of building in strict mode.
:param base_layer_dir: The path to the overlay base layer, if using overlays.
:param base_layer_hash: The validation hash of the overlay base image, if using
overlays. The validation hash should be constant for a given image, and should
change if a different base image is used.
:param project_vars_part_name: Project variables can only be set in the part
matching this name.
:param project_vars: A dictionary containing project variables.
:param partitions: A list of partitions to use when the partitions feature is
enabled. The first partition must be "default". Partitions may have an
optional namespace prefix separated by a forward slash. Partition names
must contain one or more lowercase alphanumeric characters or hyphens
("-"), and may not begin or end with a hyphen. Namespace names must
consist of only lowercase alphanumeric characters.
:param custom_args: Any additional arguments that will be passed directly
to callbacks.
"""
def __init__( # noqa: PLR0913
self,
all_parts: dict[str, Any],
*,
application_name: str,
cache_dir: Path | str,
work_dir: Path | str = ".",
arch: str = "",
base: str = "",
project_name: str | None = None,
parallel_build_count: int = 1,
application_package_name: str | None = None,
ignore_local_sources: list[str] | None = None,
extra_build_packages: list[str] | None = None,
extra_build_snaps: list[str] | None = None,
track_stage_packages: bool = False,
strict_mode: bool = False,
base_layer_dir: Path | None = None,
base_layer_hash: bytes | None = None,
project_vars_part_name: str | None = None,
project_vars: dict[str, str] | None = None,
partitions: list[str] | None = None,
**custom_args: Any, # custom passthrough args
) -> None:
# pylint: disable=too-many-locals
if not re.match("^[A-Za-z][0-9A-Za-z_]*$", application_name):
raise errors.InvalidApplicationName(application_name)
if not isinstance(all_parts, dict):
raise TypeError("parts definition must be a dictionary")
if not application_package_name:
application_package_name = application_name
if "parts" not in all_parts:
raise ValueError("parts definition is missing")
validate_partition_names(partitions)
packages.Repository.configure(application_package_name)
project_dirs = ProjectDirs(work_dir=work_dir, partitions=partitions)
project_info = ProjectInfo(
application_name=application_name,
cache_dir=Path(cache_dir),
arch=arch,
base=base,
parallel_build_count=parallel_build_count,
strict_mode=strict_mode,
project_name=project_name,
project_dirs=project_dirs,
project_vars_part_name=project_vars_part_name,
project_vars=project_vars,
partitions=partitions,
base_layer_dir=base_layer_dir,
base_layer_hash=base_layer_hash,
**custom_args,
)
parts_data = all_parts.get("parts", {})
executor.expand_environment(parts_data, info=project_info)
part_list = []
for name, spec in parts_data.items():
part = _build_part(name, spec, project_dirs, strict_mode, partitions)
_validate_part_dependencies(part, parts_data)
part_list.append(part)
self._has_overlay = any(p.has_overlay for p in part_list)
self._needs_chisel = any(p.has_slices for p in part_list)
self._has_chisel = any(p.has_chisel_as_build_snap for p in part_list)
# add a chisel as a build snap if needed
if self._needs_chisel and not self._has_chisel:
if extra_build_snaps is None:
extra_build_snaps = []
extra_build_snaps.append("chisel/latest/stable")
# a base layer is mandatory if overlays are in use
if self._has_overlay:
_ensure_overlay_supported()
if not base_layer_dir:
raise ValueError("base_layer_dir must be specified if using overlays")
if not base_layer_hash:
raise ValueError("base_layer_hash must be specified if using overlays")
else:
base_layer_dir = None
if base_layer_hash:
layer_hash: LayerHash | None = LayerHash(base_layer_hash)
else:
layer_hash = None
self._part_list = part_list
self._application_name = application_name
self._target_arch = project_info.target_arch
self._sequencer = sequencer.Sequencer(
part_list=self._part_list,
project_info=project_info,
ignore_outdated=ignore_local_sources,
base_layer_hash=layer_hash,
)
self._executor = executor.Executor(
part_list=self._part_list,
project_info=project_info,
ignore_patterns=ignore_local_sources,
extra_build_packages=extra_build_packages,
extra_build_snaps=extra_build_snaps,
track_stage_packages=track_stage_packages,
base_layer_dir=base_layer_dir,
base_layer_hash=layer_hash,
)
self._project_info = project_info
# pylint: enable=too-many-locals
@property
def project_info(self) -> ProjectInfo:
"""Obtain information about this project."""
return self._project_info
[docs]
def clean(
self, step: Step = Step.PULL, *, part_names: list[str] | None = None
) -> None:
"""Clean the specified step and parts.
Cleaning a step removes its state and all artifacts generated in that
step and subsequent steps for the specified parts.
:param step: The step to clean. If not specified, all steps will be
cleaned.
:param part_names: The list of part names to clean. If not specified,
all parts will be cleaned and work directories will be removed.
"""
self._executor.clean(initial_step=step, part_names=part_names)
[docs]
def refresh_packages_list(self) -> None:
"""Update the available packages list.
The list of available packages should be updated before planning the
sequence of actions to take. To ensure consistency between the scenarios,
it shouldn't be updated between planning and execution.
"""
packages.Repository.refresh_packages_list()
[docs]
def plan(
self,
target_step: Step,
part_names: Sequence[str] | None = None,
*,
rerun: bool = False,
) -> list[Action]:
"""Obtain the list of actions to be executed given the target step and parts.
:param target_step: The final step we want to reach.
:param part_names: The list of parts to process. If not specified, all
parts will be processed.
:return: The list of :class:`Action` objects that should be executed in
order to reach the target step for the specified parts.
"""
return self._sequencer.plan(target_step, part_names, rerun=rerun)
[docs]
def reload_state(self) -> None:
"""Reload the ephemeral state from disk."""
self._sequencer.reload_state()
[docs]
def action_executor(self) -> executor.ExecutionContext:
"""Return a context manager for action execution."""
return executor.ExecutionContext(executor=self._executor)
[docs]
def get_pull_assets(self, *, part_name: str) -> dict[str, Any] | None:
"""Return the part's pull state assets.
:param part_name: The name of the part to get assets from.
:return: The dictionary of the part's pull assets, or None if no state found.
"""
part = part_by_name(part_name, self._part_list)
state = cast(states.PullState, states.load_step_state(part, Step.PULL))
return state.assets if state else None
[docs]
def get_primed_stage_packages(self, *, part_name: str) -> list[str] | None:
"""Return the list of primed stage packages.
:param part_name: The name of the part to get primed stage packages from.
:return: The sorted list of primed stage packages, or None if no state found.
"""
part = part_by_name(part_name, self._part_list)
state = cast(states.PrimeState, states.load_step_state(part, Step.PRIME))
if not state:
return None
return sorted(state.primed_stage_packages)
def _ensure_overlay_supported() -> None:
"""Overlay is only supported in Linux and requires superuser privileges."""
if not Features().enable_overlay:
raise errors.FeatureError("Overlays are not supported.")
if sys.platform != "linux":
raise errors.OverlayPlatformError
if os.geteuid() != 0:
raise errors.OverlayPermissionError
def _build_part(
name: str,
spec: dict[str, Any],
project_dirs: ProjectDirs,
strict_plugins: bool, # noqa: FBT001
partitions: list[str] | None,
) -> Part:
"""Create and populate a :class:`Part` object based on part specification data.
:param spec: A dictionary containing the part specification.
:param project_dirs: The project's work directories.
:return: A :class:`Part` object corresponding to the given part specification.
"""
if not isinstance(spec, dict):
raise errors.PartSpecificationError(
part_name=name, message="part definition is malformed"
)
plugin_name = spec.get("plugin", "")
# If the plugin was not specified, use the part name as the plugin name.
part_name_as_plugin_name = not plugin_name
if part_name_as_plugin_name:
plugin_name = name
try:
plugin_class = plugins.get_plugin_class(plugin_name)
except ValueError as err:
if part_name_as_plugin_name:
# If plugin was not specified, avoid raising an exception telling
# that part name is an invalid plugin.
raise errors.UndefinedPlugin(part_name=name) from err
raise errors.InvalidPlugin(plugin_name, part_name=name) from err
if strict_plugins and not plugin_class.supports_strict_mode:
raise errors.PluginNotStrict(plugin_name, part_name=name)
# validate and unmarshal plugin properties
try:
properties = plugin_class.properties_class.unmarshal(spec)
except ValidationError as err:
raise errors.PartSpecificationError.from_validation_error(
part_name=name, error_list=err.errors()
) from err
except ValueError as err:
raise errors.PartSpecificationError(part_name=name, message=str(err)) from err
part_spec = plugins.extract_part_properties(spec, plugin_name=plugin_name)
# initialize part and unmarshal part specs
return Part(
name,
part_spec,
project_dirs=project_dirs,
plugin_properties=properties,
partitions=partitions,
)
def _validate_part_dependencies(part: Part, parts_data: dict[str, Any]) -> None:
for name in part.dependencies:
if name not in parts_data:
raise errors.InvalidPartName(name)