Source code for craft_parts.errors

# -*- Mode:Python; indent-tabs-mode:nil; tab-width:4 -*-
#
# Copyright 2021-2025 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/>.

"""Craft parts errors."""

import abc
import contextlib
import pathlib
from collections.abc import Iterable
from io import StringIO
from typing import TYPE_CHECKING

from overrides import override

if TYPE_CHECKING:
    from pydantic.error_wrappers import ErrorDict, Loc


[docs] class PartsError(Exception): """Unexpected error. :param brief: Brief description of error. :param details: Detailed information. :param resolution: Recommendation, if any. :param doc_slug: Reusable documentation slug for consumers adopting the Craft Parts documentation. """ def __init__( self, brief: str, details: str | None = None, resolution: str | None = None, doc_slug: str | None = None, ) -> None: self.brief = brief self._details = details self.resolution = resolution self.doc_slug = doc_slug def __str__(self) -> str: components = [self.brief] if self.details: components.append(self.details) if self.resolution: components.append(self.resolution) return "\n".join(components) def __repr__(self) -> str: return f"{self.__class__.__name__}(brief={self.brief!r}, details={self.details!r}, resolution={self.resolution!r}, doc_slug={self.doc_slug!r})" @property def details(self) -> str | None: """Further details on the error.""" return self._details
[docs] class FeatureError(PartsError): """A feature is not configured as expected.""" def __init__(self, message: str, details: str | None = None) -> None: self.message = message brief = message resolution = "This operation cannot be executed." super().__init__(brief=brief, details=details, resolution=resolution)
[docs] class PartDependencyCycle(PartsError): """A dependency cycle has been detected in the parts definition.""" def __init__(self) -> None: brief = "A circular dependency chain was detected." resolution = "Review the parts definition to remove dependency cycles." super().__init__(brief=brief, resolution=resolution)
[docs] class InvalidApplicationName(PartsError): """The application name contains invalid characters. :param name: The invalid application name. """ def __init__(self, name: str) -> None: self.name = name brief = f"Application name {name!r} is invalid." resolution = ( "Valid application names contain letters, underscores or numbers, " "and must start with a letter." ) super().__init__(brief=brief, resolution=resolution)
[docs] class InvalidPartName(PartsError): """An operation was requested on a part that's not in the parts specification. :param part_name: The invalid part name. """ def __init__(self, part_name: str) -> None: self.part_name = part_name brief = f"A part named {part_name!r} is not defined in the parts list." resolution = "Review the parts definition and make sure it's correct." super().__init__(brief=brief, resolution=resolution)
[docs] class InvalidArchitecture(PartsError): """The machine architecture is not supported. :param arch_name: The unsupported architecture name. """ def __init__(self, arch_name: str) -> None: self.arch_name = arch_name brief = f"Architecture {arch_name!r} is not supported." resolution = "Make sure the architecture name is correct." super().__init__(brief=brief, resolution=resolution)
[docs] class PartSpecificationError(PartsError): """A part was not correctly specified. :param part_name: The name of the part being processed. :param message: The error message. """ def __init__(self, *, part_name: str, message: str) -> None: self.part_name = part_name self.message = message brief = f"Part {part_name!r} validation failed." details = message resolution = f"Review part {part_name!r} and make sure it's correct." super().__init__(brief=brief, details=details, resolution=resolution)
[docs] @classmethod def from_validation_error( cls, *, part_name: str, error_list: list["ErrorDict"] ) -> "PartSpecificationError": """Create a PartSpecificationError from a pydantic error list. :param part_name: The name of the part being processed. :param error_list: A list of dictionaries containing pydantic error definitions. """ formatted_errors: list[str] = [] for error in error_list: loc = error.get("loc") msg = error.get("msg") if not (loc and msg) or not isinstance(loc, tuple): continue field = cls._format_loc(loc) if msg == "field required": formatted_errors.append(f"- field {field!r} is required") elif msg == "extra fields not permitted": formatted_errors.append(f"- extra field {field!r} not permitted") else: formatted_errors.append(f"- {msg} in field {field!r}") return cls(part_name=part_name, message="\n".join(formatted_errors))
@classmethod def _format_loc(cls, loc: "Loc") -> str: """Format location.""" loc_parts = [] for loc_part in loc: if isinstance(loc_part, str): loc_parts.append(loc_part) elif isinstance(loc_part, int): # Integer indicates an index. Go back and fix up previous part. previous_part = loc_parts.pop() previous_part += f"[{loc_part}]" loc_parts.append(previous_part) else: raise TypeError(f"unhandled loc: {loc_part}") loc_str = ".".join(loc_parts) # Filter out internal __root__ detail. return loc_str.replace(".__root__", "")
[docs] class CopyTreeError(PartsError): """Failed to copy or link a file tree. :param message: The error message. """ def __init__(self, message: str) -> None: self.message = message brief = f"Failed to copy or link file tree: {message}." resolution = "Make sure paths and permissions are correct." super().__init__(brief=brief, resolution=resolution)
[docs] class CopyFileNotFound(PartsError): """An attempt was made to copy a file that doesn't exist. :param name: The file name. """ def __init__(self, name: str) -> None: self.name = name brief = f"Failed to copy {name!r}: no such file or directory." super().__init__(brief=brief)
[docs] class XAttributeError(PartsError): """Failed to read or write an extended attribute. :param action: The action being performed. :param key: The extended attribute key. :param path: The file path. :param is_write: Whether this is an attribute write operation. """ def __init__( self, key: str, path: str, is_write: bool = False # noqa: FBT001, FBT002 ) -> None: self.key = key self.path = path self.is_write = is_write action = "write" if is_write else "read" brief = f"Unable to {action} extended attribute." details = f"Failed to {action} attribute {key!r} on {path!r}." resolution = "Make sure your filesystem supports extended attributes." super().__init__(brief=brief, details=details, resolution=resolution)
[docs] class XAttributeTooLong(PartsError): """Failed to write an extended attribute because key and/or value is too long. :param key: The extended attribute key. :param value: The extended attribute value. :param path: The file path. """ def __init__(self, key: str, value: str, path: str) -> None: self.key = key self.value = value self.path = path brief = "Failed to write attribute: key and/or value is too long." details = f"key={key!r}, value={value!r}" super().__init__(brief=brief, details=details)
[docs] class UndefinedPlugin(PartsError): """The part didn't define a plugin and the part name is not a valid plugin name. :param part_name: The name of the part with no plugin definition. """ def __init__(self, *, part_name: str) -> None: self.part_name = part_name brief = f"Plugin not defined for part {part_name!r}." resolution = f"Review part {part_name!r} and make sure it's correct." super().__init__(brief=brief, resolution=resolution)
[docs] class InvalidPlugin(PartsError): """A request was made to use a plugin that's not registered. :param plugin_name: The invalid plugin name." :param part_name: The name of the part defining the invalid plugin. """ def __init__(self, plugin_name: str, *, part_name: str) -> None: self.plugin_name = plugin_name self.part_name = part_name brief = f"Plugin {plugin_name!r} in part {part_name!r} is not registered." resolution = f"Review part {part_name!r} and make sure it's correct." super().__init__(brief=brief, resolution=resolution)
[docs] class PluginNotStrict(PartsError): """A request was made to use a plugin that's not strict. :param plugin_name: The plugin name. :param part_name: The name of the part defining the plugin. """ def __init__(self, plugin_name: str, *, part_name: str) -> None: self.plugin_name = plugin_name self.part_name = part_name brief = f"Plugin {plugin_name!r} in part {part_name!r} cannot be used." details = ( "Only plugins that are capable of building in strict mode are allowed." ) super().__init__(brief=brief, details=details)
[docs] class OsReleaseIdError(PartsError): """Failed to determine the host operating system identification string.""" def __init__(self) -> None: brief = "Unable to determine the host operating system ID." super().__init__(brief=brief)
[docs] class OsReleaseNameError(PartsError): """Failed to determine the host operating system name.""" def __init__(self) -> None: brief = "Unable to determine the host operating system name." super().__init__(brief=brief)
[docs] class OsReleaseVersionIdError(PartsError): """Failed to determine the host operating system version.""" def __init__(self) -> None: brief = "Unable to determine the host operating system version ID." super().__init__(brief=brief)
[docs] class OsReleaseCodenameError(PartsError): """Failed to determine the host operating system version codename.""" def __init__(self) -> None: brief = "Unable to determine the host operating system codename." super().__init__(brief=brief)
[docs] class FilesetError(PartsError): """An invalid fileset operation was performed. :param name: The name of the fileset. :param message: The error message. """ def __init__(self, *, name: str, message: str) -> None: self.name = name self.message = message brief = f"{name!r} fileset error: {message}." resolution = "Review the parts definition and make sure it's correct." super().__init__(brief=brief, resolution=resolution)
[docs] class FilesetConflict(PartsError): """Inconsistent stage to prime filtering. :param conflicting_files: A set containing the conflicting file names. """ def __init__(self, conflicting_files: set[str]) -> None: self.conflicting_files = conflicting_files brief = "Failed to filter files: inconsistent 'stage' and 'prime' filesets." details = ( f"The following files have been excluded in the 'stage' fileset, " f"but included by the 'prime' fileset: {conflicting_files!r}." ) resolution = ( "Make sure that the files included in 'prime' are also included in 'stage'." ) super().__init__(brief=brief, details=details, resolution=resolution)
[docs] class FileOrganizeError(PartsError): """Failed to organize a file layout. :param part_name: The name of the part being processed. :param message: The error message. """ def __init__(self, *, part_name: str, message: str) -> None: self.part_name = part_name self.message = message brief = f"Failed to organize part {part_name!r}: {message}." super().__init__(brief=brief)
[docs] class PartFilesConflict(PartsError): """Different parts list the same files with different contents. :param part_name: The name of the part being processed. :param other_part_name: The name of the conflicting part. :param conflicting_files: The list of conflicting files. :param partition: Optional name of the partition where the conflict occurred. """ def __init__( self, *, part_name: str, other_part_name: str, conflicting_files: list[str], partition: str | None = None, ) -> None: self.part_name = part_name self.other_part_name = other_part_name self.conflicting_files = conflicting_files self.partition = partition indented_conflicting_files = (f" {i}" for i in conflicting_files) file_paths = "\n".join(sorted(indented_conflicting_files)) partition_info = f" for the {partition!r} partition" if partition else "" brief = ( "Failed to stage: parts list the same file " "with different contents or permissions." ) details = ( f"Parts {part_name!r} and {other_part_name!r} list the following " f"files{partition_info}, but with different contents or permissions:\n" f"{file_paths}" ) super().__init__(brief=brief, details=details)
[docs] class StageFilesConflict(PartsError): """Files from a part conflict with files already being staged. :param part_name: The name of the part being processed. :param conflicting_files: The list of confictling files. """ def __init__(self, *, part_name: str, conflicting_files: list[str]) -> None: self.part_name = part_name self.conflicting_files = conflicting_files indented_conflicting_files = (f" {i}" for i in conflicting_files) file_paths = "\n".join(sorted(indented_conflicting_files)) brief = "Failed to stage: part files conflict with files already being staged." details = ( f"The following files in part {part_name!r} are already being staged " f"with different content:\n" f"{file_paths}" ) super().__init__(brief=brief, details=details)
[docs] class PluginEnvironmentValidationError(PartsError): """Plugin environment validation failed at runtime. :param part_name: The name of the part being processed. """ def __init__(self, *, part_name: str, reason: str) -> None: self.part_name = part_name self.reason = reason brief = f"Environment validation failed for part {part_name!r}: {reason}." super().__init__(brief=brief)
[docs] class PluginPullError(PartsError): """Plugin pull script failed at runtime. :param part_name: The name of the part being processed. """ def __init__(self, *, part_name: str) -> None: self.part_name = part_name brief = f"Failed to run the pull script for part {part_name!r}." super().__init__(brief=brief)
[docs] class UserExecutionError(PartsError, abc.ABC): """Plugin build script failed at runtime. :param part_name: The name of the part being processed. :param plugin_name: The name of the plugin being processed. """ def __init__( self, *, brief: str, resolution: str, stderr: bytes | None = None ) -> None: self.stderr = stderr super().__init__( brief=brief, resolution=resolution, doc_slug="/reference/plugins.html" ) @property @override def details(self) -> str | None: """Further details on the error. Discards all trace lines that come before the last-executed script line """ with contextlib.closing(StringIO()) as details_io: if self.stderr is None: return None stderr = self.stderr.decode("utf-8", errors="replace") stderr_lines = stderr.split("\n") # Find the final command captured in the logs last_command = None for idx, line in enumerate(reversed(stderr_lines)): if line.startswith("+"): last_command = len(stderr_lines) - idx - 1 break else: # Fallback to printing the whole log last_command = 0 for line in stderr_lines[last_command:]: if line: details_io.write(f"\n:: {line}") return details_io.getvalue()
[docs] class PluginBuildError(UserExecutionError): """Plugin build script failed at runtime. :param part_name: The name of the part being processed. :param plugin_name: The name of the plugin being processed. :param stderr: The contents of the build execution error. """ def __init__( self, *, part_name: str, plugin_name: str, stderr: bytes | None = None ) -> None: self.part_name = part_name self.plugin_name = plugin_name brief = f"Failed to run the build script for part {part_name!r}." resolution = f"Check the build output and verify the project can work with the {plugin_name!r} plugin." super().__init__(brief=brief, resolution=resolution, stderr=stderr)
[docs] class PluginCleanError(PartsError): """Script to clean strict build preparation failed at runtime. :param part_name: The name of the part being processed. """ def __init__(self, *, part_name: str) -> None: self.part_name = part_name brief = f"Failed to run the clean script for part {part_name!r}." super().__init__(brief=brief)
[docs] class InvalidControlAPICall(PartsError): """A control API call was made with invalid parameters. :param part_name: The name of the part being processed. :param scriptlet_name: The name of the scriptlet that originated the call. :param message: The error message. """ def __init__(self, *, part_name: str, scriptlet_name: str, message: str) -> None: self.part_name = part_name self.scriptlet_name = scriptlet_name self.message = message brief = ( f"{scriptlet_name!r} in part {part_name!r} executed an invalid control " f"API call: {message}." ) resolution = "Review the scriptlet and make sure it's correct." super().__init__(brief=brief, resolution=resolution)
[docs] class ScriptletRunError(UserExecutionError): """A scriptlet execution failed. :param part_name: The name of the part being processed. :param scriptlet_name: The name of the scriptlet that failed to execute. :param exit_code: The execution error code. :param stderr: The contents of the scriptlet execution error. """ def __init__( self, *, part_name: str, scriptlet_name: str, exit_code: int, stderr: bytes | None = None, ) -> None: self.part_name = part_name self.scriptlet_name = scriptlet_name self.exit_code = exit_code brief = ( f"{scriptlet_name!r} in part {part_name!r} failed with code {exit_code}." ) resolution = "Review the scriptlet and make sure it's correct." super().__init__(brief=brief, resolution=resolution, stderr=stderr)
[docs] class CallbackRegistrationError(PartsError): """Error in callback function registration. :param message: the error message. """ def __init__(self, message: str) -> None: self.message = message brief = f"Callback registration error: {message}." super().__init__(brief=brief)
[docs] class StagePackageNotFound(PartsError): """Failed to install a stage package. :param part_name: The name of the part being processed. :param package_name: The name of the package. """ def __init__(self, *, part_name: str, package_name: str) -> None: self.part_name = part_name self.package_name = package_name brief = f"Stage package not found in part {part_name!r}: {package_name}." super().__init__(brief=brief)
[docs] class OverlayPackageNotFound(PartsError): """Failed to install an overlay package. :param part_name: The name of the part being processed. :param message: the error message. """ def __init__(self, *, part_name: str, package_name: str) -> None: self.part_name = part_name self.package_name = package_name brief = f"Overlay package not found in part {part_name!r}: {package_name}." super().__init__(brief=brief)
[docs] class InvalidAction(PartsError): """An attempt was made to execute an action with invalid parameters. :param message: The error message. """ def __init__(self, message: str) -> None: self.message = message brief = f"Action is invalid: {message}." super().__init__(brief=brief)
[docs] class OverlayPlatformError(PartsError): """A project using overlays was processed on a non-Linux platform.""" def __init__(self) -> None: brief = "The overlay step is only supported on Linux." super().__init__(brief=brief)
[docs] class OverlayPermissionError(PartsError): """A project using overlays was processed by a non-privileged user.""" def __init__(self) -> None: brief = "Using the overlay step requires superuser privileges." super().__init__(brief=brief)
[docs] class DebError(PartsError): """A "deb"-related command failed.""" def __init__( self, deb_path: pathlib.Path, command: list[str], exit_code: int ) -> None: brief = ( f"Failed when handling {deb_path}: " f"command {command!r} exited with code {exit_code}." ) resolution = "Make sure the deb file is correctly specified." super().__init__(brief=brief, resolution=resolution)
[docs] class PartitionError(PartsError): """Errors related to partitions.""" def __init__( self, brief: str, *, details: str | None = None, resolution: str | None = None, ) -> None: super().__init__(brief=brief, details=details, resolution=resolution)
[docs] class PartitionUsageError(PartitionError): """Error for a list of invalid partition usages. :param error_list: Iterable of strings describing the invalid usages. :param partitions: Iterable of the names of valid partitions. :param brief: Override brief message. """ def __init__( self, error_list: Iterable[str], partitions: Iterable[str] | None, brief: str | None = None, ) -> None: valid_partitions = ( f"\nValid partitions: {', '.join(partitions)}" if partitions else "" ) super().__init__( brief=brief or "Invalid usage of partitions", details="\n".join(error_list) + valid_partitions, resolution="Correct the invalid partition name(s) and try again.", )
[docs] class PartitionUsageWarning(PartitionError, Warning): """Warnings for possibly invalid usages of partitions. :param warning_list: Iterable of strings describing the misuses. """ def __init__(self, warning_list: Iterable[str]) -> None: super().__init__( brief="Possible misuse of partitions", details=( "The following entries begin with a valid partition name but are " "not wrapped in parentheses. These entries will go into the " "default partition.\n" + "\n".join(warning_list) ), resolution=( "Wrap the partition name in parentheses, for example " "'default/file' should be written as '(default)/file'" ), ) Warning.__init__(self)
[docs] class PartitionNotFound(PartitionUsageError): """A partition has been specified that does not exist. :param partition_name: The name of the partition that does not exist. :param partitions: Iterable of the names of valid partitions. """ def __init__(self, partition_name: str, partitions: Iterable[str]) -> None: # Allow callers catching this exception easy access to the partition name self.partition_name = partition_name super().__init__( brief=f"Requested partition does not exist: {partition_name!r}", partitions=partitions, error_list=[], )