From 3b56433634709d8b318c9f538fecdf3240ae67dd Mon Sep 17 00:00:00 2001 From: Callahan Kovacs Date: Thu, 10 Oct 2024 16:54:49 -0500 Subject: [PATCH 1/4] feat: add init command Co-authored-by: Dariusz Duda Signed-off-by: Callahan Kovacs --- craft_application/commands/__init__.py | 13 +- craft_application/commands/init.py | 215 +++++++++++ craft_application/commands/other.py | 2 +- craft_application/util/__init__.py | 2 + craft_application/util/file.py | 37 ++ tests/integration/commands/__init__.py | 0 tests/integration/commands/test_init.py | 491 ++++++++++++++++++++++++ 7 files changed, 752 insertions(+), 8 deletions(-) create mode 100644 craft_application/commands/init.py create mode 100644 craft_application/util/file.py create mode 100644 tests/integration/commands/__init__.py create mode 100644 tests/integration/commands/test_init.py diff --git a/craft_application/commands/__init__.py b/craft_application/commands/__init__.py index 6f9d065a..d636c58a 100644 --- a/craft_application/commands/__init__.py +++ b/craft_application/commands/__init__.py @@ -15,17 +15,16 @@ # along with this program. If not, see . """Command classes for a craft application.""" -from craft_application.commands.base import AppCommand, ExtensibleCommand -from craft_application.commands import lifecycle -from craft_application.commands.lifecycle import ( - get_lifecycle_command_group, - LifecycleCommand, -) -from craft_application.commands.other import get_other_command_group +from .base import AppCommand, ExtensibleCommand +from . import lifecycle +from .init import InitCommand +from .lifecycle import get_lifecycle_command_group, LifecycleCommand +from .other import get_other_command_group __all__ = [ "AppCommand", "ExtensibleCommand", + "InitCommand", "lifecycle", "LifecycleCommand", "get_lifecycle_command_group", diff --git a/craft_application/commands/init.py b/craft_application/commands/init.py new file mode 100644 index 00000000..99cf7387 --- /dev/null +++ b/craft_application/commands/init.py @@ -0,0 +1,215 @@ +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . +"""Command to initialise a project.""" + +from __future__ import annotations + +import argparse +import os +import pathlib +import shutil +from textwrap import dedent +from typing import Any, cast + +from craft_cli import emit +from jinja2 import ( + Environment, + PackageLoader, + StrictUndefined, +) + +from craft_application.util import make_executable + +from . import base + + +class InitCommand(base.AppCommand): + """Command to create initial project files.""" + + name = "init" + help_msg = "Create an initial project filetree" + overview = dedent( + """ + Initialise a project. + + If is provided, initialise in that directory, + otherwise initialise in the current working directory. + """ + ) + common = True + + def fill_parser(self, parser: argparse.ArgumentParser) -> None: + """Specify command's specific parameters.""" + parser.add_argument( + "project_dir", + type=pathlib.Path, + nargs="?", + default=None, + help="Path to initialize project in; defaults to current working directory.", + ) + parser.add_argument( + "--name", + type=str, + default=None, + help="The name of project; defaults to the name of ", + ) + + def run(self, parsed_args: argparse.Namespace) -> None: + """Run the command.""" + project_dir = self._get_project_dir(parsed_args) + name = self._get_name(parsed_args) + context = self._get_context(parsed_args) + template_dir = self._get_template_dir(parsed_args) + environment = self._get_templates_environment(template_dir) + executable_files = self._get_executable_files(parsed_args) + + self._create_project_dir(project_dir=project_dir, name=name) + self._render_project( + environment, + project_dir, + template_dir, + context, + executable_files, + ) + + def _copy_template_file( + self, + template_name: str, + template_dir: pathlib.Path, + project_dir: pathlib.Path, + ) -> None: + """Copy the non-ninja template from template_dir to project_dir. + + If the file already exists in the projects copying is skipped. + + :param project_dir: The directory to render the files into. + :param template_dir: The directory where templates are stored. + :param template_name: Name of the template to copy. + """ + emit.debug(f"Copying file {template_name} to {project_dir}") + template_file = template_dir / template_name + destination_file = project_dir / template_name + if destination_file.exists(): + emit.trace(f"Skipping file {template_name} as it is already present") + return + destination_file.parent.mkdir(parents=True, exist_ok=True) + shutil.copy2(template_file, destination_file, follow_symlinks=False) + + def _render_project( + self, + environment: Environment, + project_dir: pathlib.Path, + template_dir: pathlib.Path, + context: dict[str, Any], + executable_files: list[str], + ) -> None: + """Render files for a project from a template. + + :param environment: The Jinja environment to use. + :param project_dir: The directory to render the files into. + :param template_dir: The directory where templates are stored. + :param context: The context to render the templates with. + :param executable_files: The list of files that should be executable. + """ + for template_name in environment.list_templates(): + if not template_name.endswith(".j2"): + self._copy_template_file(template_name, template_dir, project_dir) + continue + template = environment.get_template(template_name) + + # trim off `.j2` + rendered_template_name = pathlib.Path(template_name).stem + emit.debug(f"Rendering {template_name} to {rendered_template_name}") + + path = project_dir / rendered_template_name + if path.exists(): + emit.trace(f"Skipping file {template_name} as it is already present") + continue + path.parent.mkdir(parents=True, exist_ok=True) + with path.open("wt", encoding="utf8") as file: + out = template.render(context) + file.write(out) + if rendered_template_name in executable_files and os.name == "posix": + make_executable(file) + emit.debug(" made executable") + emit.message("Successfully initialised project.") + + def _get_template_dir( + self, + parsed_args: argparse.Namespace, # noqa: ARG002 (unused-method-argument) + ) -> pathlib.Path: + """Return the path to the template directory.""" + return pathlib.Path("templates") + + def _get_executable_files( + self, + parsed_args: argparse.Namespace, # noqa: ARG002 (unused-method-argument) + ) -> list[str]: + """Return the list of files that should be executable.""" + return [] + + def _get_context(self, parsed_args: argparse.Namespace) -> dict[str, Any]: + """Get context to render templates with. + + :returns: A dict of context variables. + """ + name = self._get_name(parsed_args) + emit.debug(f"Set project name to '{name}'") + + return {"name": name} + + def _get_project_dir(self, parsed_args: argparse.Namespace) -> pathlib.Path: + """Get project dir where project should be initialized. + + It applies rules in the following order: + - if is specified explicitly, it returns + - if is undefined, it defaults to cwd + """ + # if set explicitly, just return it + if parsed_args.project_dir is not None: + return pathlib.Path(parsed_args.project_dir) + + # If both args are undefined, default to current dir + return pathlib.Path.cwd().resolve() + + def _get_name(self, parsed_args: argparse.Namespace) -> str: + """Get name of the package that is about to be initialized. + + Check if name is set explicitly or fallback to project_dir. + """ + if parsed_args.name is not None: + return cast(str, parsed_args.name) + return self._get_project_dir(parsed_args).name + + def _create_project_dir( + self, + project_dir: pathlib.Path, + *, + name: str, + ) -> None: + """Create the project path if it does not already exist.""" + if not project_dir.exists(): + project_dir.mkdir(parents=True) + emit.debug(f"Using project directory {str(project_dir)!r} for {name}") + + def _get_templates_environment(self, template_dir: pathlib.Path) -> Environment: + """Create and return a Jinja environment to deal with the templates.""" + loader = PackageLoader(self._app.name, str(template_dir)) + return Environment( + loader=loader, + autoescape=False, # noqa: S701 (jinja2-autoescape-false) + keep_trailing_newline=True, # they're not text files if they don't end in newline! + optimized=False, # optimization doesn't make sense for one-offs + undefined=StrictUndefined, + ) # fail on undefined diff --git a/craft_application/commands/other.py b/craft_application/commands/other.py index eb30d80f..3994dc64 100644 --- a/craft_application/commands/other.py +++ b/craft_application/commands/other.py @@ -37,7 +37,7 @@ def get_other_command_group() -> CommandGroup: class VersionCommand(base.AppCommand): - """Show the snapcraft version.""" + """Show the application version.""" name = "version" help_msg = "Show the application version and exit" diff --git a/craft_application/util/__init__.py b/craft_application/util/__init__.py index 6bd33ead..49e913e2 100644 --- a/craft_application/util/__init__.py +++ b/craft_application/util/__init__.py @@ -17,6 +17,7 @@ from craft_application.util.callbacks import get_unique_callbacks from craft_application.util.docs import render_doc_url +from craft_application.util.file import make_executable from craft_application.util.logging import setup_loggers from craft_application.util.paths import get_filename_from_url_path, get_managed_logpath from craft_application.util.platforms import ( @@ -38,6 +39,7 @@ __all__ = [ "get_unique_callbacks", "render_doc_url", + "make_executable", "setup_loggers", "get_filename_from_url_path", "get_managed_logpath", diff --git a/craft_application/util/file.py b/craft_application/util/file.py new file mode 100644 index 00000000..fccf39dd --- /dev/null +++ b/craft_application/util/file.py @@ -0,0 +1,37 @@ +# This file is part of craft_application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . + +"""Helper utilities for files.""" + +import io +import os +from _stat import S_IRGRP, S_IROTH, S_IRUSR, S_IXGRP, S_IXOTH, S_IXUSR + +# handy masks for execution and reading for everybody +S_IXALL = S_IXUSR | S_IXGRP | S_IXOTH +S_IRALL = S_IRUSR | S_IRGRP | S_IROTH + + +def make_executable(fh: io.IOBase) -> None: + """Make open file fh executable. + :param fh: An open file object. + """ + fileno = fh.fileno() + mode = os.fstat(fileno).st_mode + mode_r = mode & S_IRALL + mode_x = mode_r >> 2 + mode = mode | mode_x + os.fchmod(fileno, mode) diff --git a/tests/integration/commands/__init__.py b/tests/integration/commands/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/tests/integration/commands/test_init.py b/tests/integration/commands/test_init.py new file mode 100644 index 00000000..d8afe143 --- /dev/null +++ b/tests/integration/commands/test_init.py @@ -0,0 +1,491 @@ +# This file is part of craft-application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . +"""Tests for init command.""" + +import argparse +import sys +from pathlib import Path +from textwrap import dedent + +import pytest +from craft_application.commands.init import ( + InitCommand, +) +from craft_application.models.project import Project +from jinja2 import Environment, FileSystemLoader, StrictUndefined, Undefined + + +@pytest.fixture +def fake_empty_template_dir(tmp_path) -> Path: + empty_template_dir_path = Path(tmp_path / "fake_template_dir") + empty_template_dir_path.mkdir() + return empty_template_dir_path + + +@pytest.fixture +def project_yaml_filename() -> str: + return "testcraft.yaml" + + +@pytest.fixture +def template_dir_with_testcraft_yaml_j2( + fake_empty_template_dir: Path, + project_yaml_filename: str, +) -> Path: + (fake_empty_template_dir / f"{project_yaml_filename}.j2").write_text( + dedent( + """ + # This file configures testcraft. + + # (Required) + name: {{ name }} + + # (Required) + # The source package version + version: git + + # (Required) + # Version of the build base OS + base: ubuntu@24.04 + + # (Recommended) + title: Testcraft Template Package + + # (Required) + summary: A very short one-line summary of the package. + + # (Required) + description: | + A single sentence that says what the source is, concisely and memorably. + + A paragraph of one to three short sentences, that describe what the package does. + + A third paragraph that explains what need the package meets. + + Finally, a paragraph that describes whom the package is useful for. + + + parts: + {{ name }}: + plugin: nil + source: . + platforms: + amd64: + """ + ) + ) + + return fake_empty_template_dir + + +@pytest.fixture +def template_dir_with_multiple_non_ninja_files( + fake_empty_template_dir: Path, +) -> Path: + file_1 = fake_empty_template_dir / "file1.txt" + file_1.write_text("Content of file1.txt") + file_2 = fake_empty_template_dir / "nested" / "file2.txt" + file_2.parent.mkdir() + file_2.write_text("Content of the nested file") + return fake_empty_template_dir + + +@pytest.fixture +def template_dir_with_symlinks( + template_dir_with_testcraft_yaml_j2: Path, +) -> Path: + symlink_to_python_executable = template_dir_with_testcraft_yaml_j2 / "py3_symlink" + symlink_to_python_executable.symlink_to(sys.executable) + return template_dir_with_testcraft_yaml_j2 + + +@pytest.fixture +def fake_empty_project_dir(tmp_path, monkeypatch) -> Path: + empty_project_dir_path = Path(tmp_path / "fake-project-dir") + empty_project_dir_path.mkdir() + monkeypatch.chdir(empty_project_dir_path) + return empty_project_dir_path + + +@pytest.fixture +def non_empty_project_dir(tmp_path, monkeypatch) -> Path: + non_empty_project_dir_path = Path(tmp_path / "fake-non-empty-project-dir") + non_empty_project_dir_path.mkdir() + (non_empty_project_dir_path / "some_project_file").touch() + monkeypatch.chdir(non_empty_project_dir_path) + return non_empty_project_dir_path + + +def get_jinja2_template_environment( + fake_template_dir: Path, + *, + autoescape: bool = False, + keep_trailing_newline: bool = True, + optimized: bool = False, + undefined: type[Undefined] = StrictUndefined, +) -> Environment: + return Environment( + loader=FileSystemLoader(fake_template_dir), + autoescape=autoescape, # noqa: S701 (jinja2-autoescape-false) + keep_trailing_newline=keep_trailing_newline, + optimized=optimized, + undefined=undefined, + ) + + +def get_command_arguments( + *, + name: str | None = None, + project_dir: Path | None = None, +) -> argparse.Namespace: + return argparse.Namespace( + project_dir=project_dir, + name=name, + ) + + +@pytest.fixture +def init_command(app_metadata) -> InitCommand: + return InitCommand({"app": app_metadata, "services": []}) + + +def test_init_works_with_empty_templates_dir( + init_command: InitCommand, + fake_empty_project_dir: Path, + fake_empty_template_dir, + emitter, + mocker, + check, +): + parsed_args = get_command_arguments() + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment(fake_empty_template_dir), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + with check: + assert not list( + fake_empty_project_dir.iterdir() + ), "Project dir should be initialized empty" + + +def test_init_works_with_single_template_file( + init_command: InitCommand, + fake_empty_project_dir: Path, + project_yaml_filename: str, + template_dir_with_testcraft_yaml_j2: Path, + emitter, + mocker, + check, +): + parsed_args = get_command_arguments() + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_testcraft_yaml_j2 + ), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + project_yaml_path = fake_empty_project_dir / project_yaml_filename + + with check: + assert project_yaml_path.exists(), "Project should be initialized with template" + project = Project.from_yaml_file(project_yaml_path) + with check: + assert project.name == fake_empty_project_dir.name + + +def test_init_works_with_single_template_and_custom_name( + init_command: InitCommand, + fake_empty_project_dir: Path, + project_yaml_filename: str, + template_dir_with_testcraft_yaml_j2: Path, + emitter, + mocker, + check, +): + custom_name = "some-other-testproject" + parsed_args = get_command_arguments(name=custom_name) + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_testcraft_yaml_j2 + ), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + project_yaml_path = Path(fake_empty_project_dir, project_yaml_filename) + + with check: + assert project_yaml_path.exists(), "Project should be initialized with template" + project = Project.from_yaml_file(project_yaml_path) + with check: + assert project.name == custom_name + + +def check_file_existence_and_content( + check, + file_path: Path, + content: str, +) -> None: + with check: + assert file_path.exists(), f"{file_path.name} should be created" + + with check: + assert file_path.read_text() == content, f"{file_path.name} incorrect content" + + +def test_init_works_with_non_jinja2_templates( + init_command: InitCommand, + fake_empty_project_dir: Path, + template_dir_with_multiple_non_ninja_files: Path, + emitter, + mocker, + check, +): + parsed_args = get_command_arguments() + + mocker.patch.object( + init_command, + "_get_template_dir", + return_value=template_dir_with_multiple_non_ninja_files, + ) + + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_multiple_non_ninja_files + ), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + check_file_existence_and_content( + check, fake_empty_project_dir / "file1.txt", "Content of file1.txt" + ) + check_file_existence_and_content( + check, + fake_empty_project_dir / "nested" / "file2.txt", + "Content of the nested file", + ) + + +def test_init_does_not_follow_symlinks_but_copies_them_as_is( + init_command: InitCommand, + fake_empty_project_dir: Path, + project_yaml_filename: str, + template_dir_with_symlinks: Path, + mocker, + check, +): + parsed_args = get_command_arguments() + + mocker.patch.object( + init_command, + "_get_template_dir", + return_value=template_dir_with_symlinks, + ) + + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment(template_dir_with_symlinks), + ) + init_command.run(parsed_args) + + project = Project.from_yaml_file(fake_empty_project_dir / project_yaml_filename) + with check: + assert project.name == fake_empty_project_dir.name + with check: + assert ( + fake_empty_project_dir / "py3_symlink" + ).is_symlink(), "Symlink should be left intact." + + +def test_init_does_not_fail_on_non_empty_dir( + init_command: InitCommand, + non_empty_project_dir: Path, + project_yaml_filename: str, + template_dir_with_testcraft_yaml_j2: Path, + mocker, + emitter, + check, +): + parsed_args = get_command_arguments() + + mocker.patch.object( + init_command, + "_get_template_dir", + return_value=template_dir_with_testcraft_yaml_j2, + ) + + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_testcraft_yaml_j2 + ), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + project_yaml_path = non_empty_project_dir / project_yaml_filename + + with check: + assert project_yaml_path.exists(), "Project should be initialized with template" + project = Project.from_yaml_file(project_yaml_path) + with check: + assert project.name == non_empty_project_dir.name + + +def test_init_does_not_override_existing_craft_yaml( + init_command: InitCommand, + non_empty_project_dir: Path, + project_yaml_filename: str, + fake_project: Project, + template_dir_with_testcraft_yaml_j2: Path, + mocker, + emitter, + check, +): + parsed_args = get_command_arguments(project_dir=non_empty_project_dir) + fake_project.to_yaml_file(non_empty_project_dir / project_yaml_filename) + + mocker.patch.object( + init_command, + "_get_template_dir", + return_value=template_dir_with_testcraft_yaml_j2, + ) + + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_testcraft_yaml_j2 + ), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + project_yaml_path = non_empty_project_dir / project_yaml_filename + + with check: + assert project_yaml_path.exists(), "Project should be initialized with template" + project = Project.from_yaml_file(project_yaml_path) + with check: + assert project.name == fake_project.name + + +def test_init_with_different_name_and_directory( + init_command: InitCommand, + fake_empty_project_dir: Path, + template_dir_with_testcraft_yaml_j2: Path, + project_yaml_filename: str, + mocker, + emitter, + check, +): + custom_project_name = "some-custom-project" + parsed_args = get_command_arguments( + name=custom_project_name, + project_dir=fake_empty_project_dir, + ) + + mocker.patch.object( + init_command, + "_get_template_dir", + return_value=template_dir_with_testcraft_yaml_j2, + ) + + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_testcraft_yaml_j2 + ), + ) + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + project_yaml_path = fake_empty_project_dir / project_yaml_filename + + with check: + assert project_yaml_path.exists(), "Project should be initialized with template" + project = Project.from_yaml_file(project_yaml_path) + with check: + assert project.name == custom_project_name + + +def test_init_with_default_arguments_uses_current_directory( + init_command: InitCommand, + fake_empty_project_dir: Path, + template_dir_with_testcraft_yaml_j2: Path, + project_yaml_filename: str, + mocker, + emitter, + check, +): + expected_project_name = fake_empty_project_dir.name + parsed_args = get_command_arguments() + + mocker.patch.object( + init_command, + "_get_template_dir", + return_value=template_dir_with_testcraft_yaml_j2, + ) + + mocker.patch.object( + init_command, + "_get_templates_environment", + return_value=get_jinja2_template_environment( + template_dir_with_testcraft_yaml_j2 + ), + ) + + init_command.run(parsed_args) + + with check: + assert emitter.assert_message("Successfully initialised project.") + + project_yaml_path = fake_empty_project_dir / project_yaml_filename + + with check: + assert project_yaml_path.exists(), "Project should be initialized with template" + project = Project.from_yaml_file(project_yaml_path) + with check: + assert project.name == expected_project_name From c6e2dd35c7c96d16c5e5c48d0856b1ef52ba7080 Mon Sep 17 00:00:00 2001 From: Callahan Kovacs Date: Fri, 11 Oct 2024 18:01:43 -0500 Subject: [PATCH 2/4] feat: create InitService Signed-off-by: Callahan Kovacs --- craft_application/commands/init.py | 150 +-------- craft_application/commands/other.py | 3 +- craft_application/services/__init__.py | 2 + craft_application/services/init.py | 155 +++++++++ craft_application/services/service_factory.py | 2 + craft_application/util/file.py | 8 +- pyproject.toml | 1 + tests/conftest.py | 11 + tests/integration/commands/__init__.py | 0 .../{commands => services}/test_init.py | 307 +++++------------- tests/integration/test_application.py | 3 +- tests/unit/commands/test_init.py | 58 ++++ tests/unit/commands/test_other.py | 7 +- tests/unit/conftest.py | 1 + tests/unit/remote/conftest.py | 29 -- tests/unit/services/test_init.py | 205 ++++++++++++ tests/unit/util/test_file.py | 54 +++ 17 files changed, 598 insertions(+), 398 deletions(-) create mode 100644 craft_application/services/init.py delete mode 100644 tests/integration/commands/__init__.py rename tests/integration/{commands => services}/test_init.py (53%) create mode 100644 tests/unit/commands/test_init.py delete mode 100644 tests/unit/remote/conftest.py create mode 100644 tests/unit/services/test_init.py create mode 100644 tests/unit/util/test_file.py diff --git a/craft_application/commands/init.py b/craft_application/commands/init.py index 99cf7387..d1bb7324 100644 --- a/craft_application/commands/init.py +++ b/craft_application/commands/init.py @@ -16,20 +16,9 @@ from __future__ import annotations import argparse -import os import pathlib -import shutil from textwrap import dedent -from typing import Any, cast - -from craft_cli import emit -from jinja2 import ( - Environment, - PackageLoader, - StrictUndefined, -) - -from craft_application.util import make_executable +from typing import cast from . import base @@ -69,107 +58,19 @@ def run(self, parsed_args: argparse.Namespace) -> None: """Run the command.""" project_dir = self._get_project_dir(parsed_args) name = self._get_name(parsed_args) - context = self._get_context(parsed_args) - template_dir = self._get_template_dir(parsed_args) - environment = self._get_templates_environment(template_dir) - executable_files = self._get_executable_files(parsed_args) - - self._create_project_dir(project_dir=project_dir, name=name) - self._render_project( - environment, - project_dir, - template_dir, - context, - executable_files, - ) - - def _copy_template_file( - self, - template_name: str, - template_dir: pathlib.Path, - project_dir: pathlib.Path, - ) -> None: - """Copy the non-ninja template from template_dir to project_dir. - - If the file already exists in the projects copying is skipped. - - :param project_dir: The directory to render the files into. - :param template_dir: The directory where templates are stored. - :param template_name: Name of the template to copy. - """ - emit.debug(f"Copying file {template_name} to {project_dir}") - template_file = template_dir / template_name - destination_file = project_dir / template_name - if destination_file.exists(): - emit.trace(f"Skipping file {template_name} as it is already present") - return - destination_file.parent.mkdir(parents=True, exist_ok=True) - shutil.copy2(template_file, destination_file, follow_symlinks=False) - - def _render_project( - self, - environment: Environment, - project_dir: pathlib.Path, - template_dir: pathlib.Path, - context: dict[str, Any], - executable_files: list[str], - ) -> None: - """Render files for a project from a template. + self._services.init.run(project_dir=project_dir, name=name) - :param environment: The Jinja environment to use. - :param project_dir: The directory to render the files into. - :param template_dir: The directory where templates are stored. - :param context: The context to render the templates with. - :param executable_files: The list of files that should be executable. - """ - for template_name in environment.list_templates(): - if not template_name.endswith(".j2"): - self._copy_template_file(template_name, template_dir, project_dir) - continue - template = environment.get_template(template_name) - - # trim off `.j2` - rendered_template_name = pathlib.Path(template_name).stem - emit.debug(f"Rendering {template_name} to {rendered_template_name}") - - path = project_dir / rendered_template_name - if path.exists(): - emit.trace(f"Skipping file {template_name} as it is already present") - continue - path.parent.mkdir(parents=True, exist_ok=True) - with path.open("wt", encoding="utf8") as file: - out = template.render(context) - file.write(out) - if rendered_template_name in executable_files and os.name == "posix": - make_executable(file) - emit.debug(" made executable") - emit.message("Successfully initialised project.") - - def _get_template_dir( - self, - parsed_args: argparse.Namespace, # noqa: ARG002 (unused-method-argument) - ) -> pathlib.Path: - """Return the path to the template directory.""" - return pathlib.Path("templates") - - def _get_executable_files( - self, - parsed_args: argparse.Namespace, # noqa: ARG002 (unused-method-argument) - ) -> list[str]: - """Return the list of files that should be executable.""" - return [] - - def _get_context(self, parsed_args: argparse.Namespace) -> dict[str, Any]: - """Get context to render templates with. + def _get_name(self, parsed_args: argparse.Namespace) -> str: + """Get name of the package that is about to be initialized. - :returns: A dict of context variables. + Check if name is set explicitly or fallback to project_dir. """ - name = self._get_name(parsed_args) - emit.debug(f"Set project name to '{name}'") - - return {"name": name} + if parsed_args.name is not None: + return cast(str, parsed_args.name) + return self._get_project_dir(parsed_args).name - def _get_project_dir(self, parsed_args: argparse.Namespace) -> pathlib.Path: + @staticmethod + def _get_project_dir(parsed_args: argparse.Namespace) -> pathlib.Path: """Get project dir where project should be initialized. It applies rules in the following order: @@ -182,34 +83,3 @@ def _get_project_dir(self, parsed_args: argparse.Namespace) -> pathlib.Path: # If both args are undefined, default to current dir return pathlib.Path.cwd().resolve() - - def _get_name(self, parsed_args: argparse.Namespace) -> str: - """Get name of the package that is about to be initialized. - - Check if name is set explicitly or fallback to project_dir. - """ - if parsed_args.name is not None: - return cast(str, parsed_args.name) - return self._get_project_dir(parsed_args).name - - def _create_project_dir( - self, - project_dir: pathlib.Path, - *, - name: str, - ) -> None: - """Create the project path if it does not already exist.""" - if not project_dir.exists(): - project_dir.mkdir(parents=True) - emit.debug(f"Using project directory {str(project_dir)!r} for {name}") - - def _get_templates_environment(self, template_dir: pathlib.Path) -> Environment: - """Create and return a Jinja environment to deal with the templates.""" - loader = PackageLoader(self._app.name, str(template_dir)) - return Environment( - loader=loader, - autoescape=False, # noqa: S701 (jinja2-autoescape-false) - keep_trailing_newline=True, # they're not text files if they don't end in newline! - optimized=False, # optimization doesn't make sense for one-offs - undefined=StrictUndefined, - ) # fail on undefined diff --git a/craft_application/commands/other.py b/craft_application/commands/other.py index 3994dc64..25963089 100644 --- a/craft_application/commands/other.py +++ b/craft_application/commands/other.py @@ -18,7 +18,7 @@ from craft_cli import CommandGroup, emit -from craft_application.commands import base +from . import InitCommand, base if TYPE_CHECKING: # pragma: no cover import argparse @@ -27,6 +27,7 @@ def get_other_command_group() -> CommandGroup: """Return the lifecycle related command group.""" commands: list[type[base.AppCommand]] = [ + InitCommand, VersionCommand, ] diff --git a/craft_application/services/__init__.py b/craft_application/services/__init__.py index accbd923..e9c066c2 100644 --- a/craft_application/services/__init__.py +++ b/craft_application/services/__init__.py @@ -19,6 +19,7 @@ from craft_application.services.config import ConfigService from craft_application.services.fetch import FetchService from craft_application.services.lifecycle import LifecycleService +from craft_application.services.init import InitService from craft_application.services.package import PackageService from craft_application.services.provider import ProviderService from craft_application.services.remotebuild import RemoteBuildService @@ -31,6 +32,7 @@ "ProjectService", "ConfigService", "LifecycleService", + "InitService", "PackageService", "ProviderService", "RemoteBuildService", diff --git a/craft_application/services/init.py b/craft_application/services/init.py new file mode 100644 index 00000000..71b016e4 --- /dev/null +++ b/craft_application/services/init.py @@ -0,0 +1,155 @@ +# This file is part of craft-application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . + +"""Service for initializing a project.""" + +import os +import pathlib +import shutil +from typing import Any + +import jinja2 +from craft_cli import emit + +from craft_application.util import make_executable + +from . import base + + +class InitService(base.AppService): + """Service class for initializing a project.""" + + def run(self, *, project_dir: pathlib.Path, name: str) -> None: + """Initialize a new project.""" + context = self._get_context(name=name) + template_dir = self._get_template_dir() + environment = self._get_templates_environment(template_dir) + executable_files = self._get_executable_files() + + self._create_project_dir(project_dir=project_dir, name=name) + self._render_project( + environment, + project_dir, + template_dir, + context, + executable_files, + ) + + def _copy_template_file( + self, + template_name: str, + template_dir: pathlib.Path, + project_dir: pathlib.Path, + ) -> None: + """Copy the non-ninja template from template_dir to project_dir. + + If the file already exists in the projects copying is skipped. + + :param project_dir: The directory to render the files into. + :param template_dir: The directory where templates are stored. + :param template_name: Name of the template to copy. + """ + emit.debug(f"Copying file {template_name} to {project_dir}") + template_file = template_dir / template_name + destination_file = project_dir / template_name + if destination_file.exists(): + emit.debug(f"Skipping file {template_name} because it is already present.") + return + destination_file.parent.mkdir(parents=True, exist_ok=True) + shutil.copy2(template_file, destination_file, follow_symlinks=False) + + def _render_project( + self, + environment: jinja2.Environment, + project_dir: pathlib.Path, + template_dir: pathlib.Path, + context: dict[str, Any], + executable_files: list[str], + ) -> None: + """Render files for a project from a template. + + :param environment: The Jinja environment to use. + :param project_dir: The directory to render the files into. + :param template_dir: The directory where templates are stored. + :param context: The context to render the templates with. + :param executable_files: The list of files that should be executable. + """ + for template_name in environment.list_templates(): + if not template_name.endswith(".j2"): + self._copy_template_file(template_name, template_dir, project_dir) + continue + template = environment.get_template(template_name) + + # trim off `.j2` + rendered_template_name = template_name[:-3] + emit.debug(f"Rendering {template_name} to {rendered_template_name}") + + path = project_dir / rendered_template_name + if path.exists(): + emit.trace(f"Skipping file {template_name} as it is already present") + continue + path.parent.mkdir(parents=True, exist_ok=True) + with path.open("wt", encoding="utf8") as file: + out = template.render(context) + file.write(out) + if rendered_template_name in executable_files and os.name == "posix": + make_executable(file) + emit.debug(" made executable") + emit.message("Successfully initialised project.") + + def _get_template_dir(self) -> pathlib.Path: + """Return the path to the template directory.""" + return pathlib.Path("templates") + + def _get_executable_files(self) -> list[str]: + """Return the list of files that should be executable.""" + return [] + + def _get_context(self, name: str) -> dict[str, Any]: + """Get context to render templates with. + + :returns: A dict of context variables. + """ + emit.debug(f"Set project name to '{name}'") + + return {"name": name} + + def _create_project_dir( + self, + project_dir: pathlib.Path, + *, + name: str, + ) -> None: + """Create the project path if it does not already exist.""" + if not project_dir.exists(): + project_dir.mkdir(parents=True) + emit.debug(f"Using project directory {str(project_dir)!r} for {name}") + + def _get_loader(self, template_dir: pathlib.Path) -> jinja2.PackageLoader: + """Return a Jinja template loader for the given template directory.""" + return jinja2.PackageLoader(self._app.name, str(template_dir)) + + def _get_templates_environment( + self, template_dir: pathlib.Path + ) -> jinja2.Environment: + """Create and return a Jinja environment to deal with the templates.""" + return jinja2.Environment( + loader=self._get_loader(template_dir), + autoescape=False, # noqa: S701 (jinja2-autoescape-false) + keep_trailing_newline=True, # they're not text files if they don't end in newline! + optimized=False, # optimization doesn't make sense for one-offs + undefined=jinja2.StrictUndefined, + ) # fail on undefined diff --git a/craft_application/services/service_factory.py b/craft_application/services/service_factory.py index d7c3cf4f..51416e52 100644 --- a/craft_application/services/service_factory.py +++ b/craft_application/services/service_factory.py @@ -44,6 +44,7 @@ class ServiceFactory: RequestClass: type[services.RequestService] = services.RequestService ConfigClass: type[services.ConfigService] = services.ConfigService FetchClass: type[services.FetchService] = services.FetchService + InitClass: type[services.InitService] = services.InitService project: models.Project | None = None @@ -57,6 +58,7 @@ class ServiceFactory: request: services.RequestService = None # type: ignore[assignment] config: services.ConfigService = None # type: ignore[assignment] fetch: services.FetchService = None # type: ignore[assignment] + init: services.InitService = None # type: ignore[assignment] def __post_init__(self) -> None: self._service_kwargs: dict[str, dict[str, Any]] = {} diff --git a/craft_application/util/file.py b/craft_application/util/file.py index fccf39dd..7a0e4e62 100644 --- a/craft_application/util/file.py +++ b/craft_application/util/file.py @@ -18,15 +18,17 @@ import io import os -from _stat import S_IRGRP, S_IROTH, S_IRUSR, S_IXGRP, S_IXOTH, S_IXUSR +from _stat import S_IRGRP, S_IROTH, S_IRUSR -# handy masks for execution and reading for everybody -S_IXALL = S_IXUSR | S_IXGRP | S_IXOTH S_IRALL = S_IRUSR | S_IRGRP | S_IROTH +"""0o444 permission mask for execution permissions for everybody.""" def make_executable(fh: io.IOBase) -> None: """Make open file fh executable. + + Only makes the file executable for the user, group, or other if they already had read permissions. + :param fh: An open file object. """ fileno = fh.fileno() diff --git a/pyproject.toml b/pyproject.toml index 41e39803..c593e4f4 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -9,6 +9,7 @@ dependencies = [ "craft-parts>=2.1.1", "craft-platforms>=0.3.1", "craft-providers>=2.0.4", + "Jinja2~=3.1", "snap-helpers>=0.4.2", "platformdirs>=3.10", "pydantic~=2.0", diff --git a/tests/conftest.py b/tests/conftest.py index 9da4a645..c198584f 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -339,3 +339,14 @@ def app(app_metadata, fake_services): @pytest.fixture def manifest_data_dir(): return pathlib.Path(__file__).parent / "data/manifest" + + +@pytest.fixture +def new_dir(tmp_path): + """Change to a new temporary directory.""" + cwd = pathlib.Path.cwd() + os.chdir(tmp_path) + + yield tmp_path + + os.chdir(cwd) diff --git a/tests/integration/commands/__init__.py b/tests/integration/commands/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/integration/commands/test_init.py b/tests/integration/services/test_init.py similarity index 53% rename from tests/integration/commands/test_init.py rename to tests/integration/services/test_init.py index d8afe143..dcad9838 100644 --- a/tests/integration/commands/test_init.py +++ b/tests/integration/services/test_init.py @@ -13,25 +13,40 @@ # # You should have received a copy of the GNU Lesser General Public License along # with this program. If not, see . -"""Tests for init command.""" -import argparse +"""Tests for init service.""" + +import pathlib import sys -from pathlib import Path -from textwrap import dedent +import textwrap import pytest -from craft_application.commands.init import ( - InitCommand, -) from craft_application.models.project import Project -from jinja2 import Environment, FileSystemLoader, StrictUndefined, Undefined +from craft_application.services import InitService +from jinja2 import FileSystemLoader + +# init operates in the current working directory +pytestmark = pytest.mark.usefixtures("new_dir") + + +@pytest.fixture +def init_service(app_metadata, fake_services, mocker, tmp_path): + _init_service = InitService(app_metadata, fake_services) + _init_service.setup() + + mocker.patch.object( + _init_service, + "_get_loader", + return_value=FileSystemLoader(tmp_path / "templates"), + ) + + return _init_service @pytest.fixture -def fake_empty_template_dir(tmp_path) -> Path: - empty_template_dir_path = Path(tmp_path / "fake_template_dir") - empty_template_dir_path.mkdir() +def fake_empty_template_dir(tmp_path) -> pathlib.Path: + empty_template_dir_path = pathlib.Path(tmp_path / "templates") + empty_template_dir_path.mkdir(parents=True) return empty_template_dir_path @@ -40,13 +55,16 @@ def project_yaml_filename() -> str: return "testcraft.yaml" +# TODO: test nested templates + + @pytest.fixture def template_dir_with_testcraft_yaml_j2( - fake_empty_template_dir: Path, + fake_empty_template_dir: pathlib.Path, project_yaml_filename: str, -) -> Path: +) -> pathlib.Path: (fake_empty_template_dir / f"{project_yaml_filename}.j2").write_text( - dedent( + textwrap.dedent( """ # This file configures testcraft. @@ -93,8 +111,8 @@ def template_dir_with_testcraft_yaml_j2( @pytest.fixture def template_dir_with_multiple_non_ninja_files( - fake_empty_template_dir: Path, -) -> Path: + fake_empty_template_dir: pathlib.Path, +) -> pathlib.Path: file_1 = fake_empty_template_dir / "file1.txt" file_1.write_text("Content of file1.txt") file_2 = fake_empty_template_dir / "nested" / "file2.txt" @@ -105,78 +123,37 @@ def template_dir_with_multiple_non_ninja_files( @pytest.fixture def template_dir_with_symlinks( - template_dir_with_testcraft_yaml_j2: Path, -) -> Path: + template_dir_with_testcraft_yaml_j2: pathlib.Path, +) -> pathlib.Path: symlink_to_python_executable = template_dir_with_testcraft_yaml_j2 / "py3_symlink" symlink_to_python_executable.symlink_to(sys.executable) return template_dir_with_testcraft_yaml_j2 @pytest.fixture -def fake_empty_project_dir(tmp_path, monkeypatch) -> Path: - empty_project_dir_path = Path(tmp_path / "fake-project-dir") +def fake_empty_project_dir(tmp_path) -> pathlib.Path: + empty_project_dir_path = pathlib.Path(tmp_path / "fake-project-dir") empty_project_dir_path.mkdir() - monkeypatch.chdir(empty_project_dir_path) return empty_project_dir_path @pytest.fixture -def non_empty_project_dir(tmp_path, monkeypatch) -> Path: - non_empty_project_dir_path = Path(tmp_path / "fake-non-empty-project-dir") +def non_empty_project_dir(tmp_path) -> pathlib.Path: + non_empty_project_dir_path = pathlib.Path(tmp_path / "fake-non-empty-project-dir") non_empty_project_dir_path.mkdir() (non_empty_project_dir_path / "some_project_file").touch() - monkeypatch.chdir(non_empty_project_dir_path) return non_empty_project_dir_path -def get_jinja2_template_environment( - fake_template_dir: Path, - *, - autoescape: bool = False, - keep_trailing_newline: bool = True, - optimized: bool = False, - undefined: type[Undefined] = StrictUndefined, -) -> Environment: - return Environment( - loader=FileSystemLoader(fake_template_dir), - autoescape=autoescape, # noqa: S701 (jinja2-autoescape-false) - keep_trailing_newline=keep_trailing_newline, - optimized=optimized, - undefined=undefined, - ) - - -def get_command_arguments( - *, - name: str | None = None, - project_dir: Path | None = None, -) -> argparse.Namespace: - return argparse.Namespace( - project_dir=project_dir, - name=name, - ) - - -@pytest.fixture -def init_command(app_metadata) -> InitCommand: - return InitCommand({"app": app_metadata, "services": []}) - - +@pytest.mark.usefixtures("fake_empty_template_dir") def test_init_works_with_empty_templates_dir( - init_command: InitCommand, - fake_empty_project_dir: Path, - fake_empty_template_dir, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, emitter, - mocker, check, ): - parsed_args = get_command_arguments() - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment(fake_empty_template_dir), - ) - init_command.run(parsed_args) + """Initialize a project with an empty templates directory.""" + init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") with check: assert emitter.assert_message("Successfully initialised project.") @@ -186,24 +163,16 @@ def test_init_works_with_empty_templates_dir( ), "Project dir should be initialized empty" +@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_works_with_single_template_file( - init_command: InitCommand, - fake_empty_project_dir: Path, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, project_yaml_filename: str, - template_dir_with_testcraft_yaml_j2: Path, emitter, - mocker, check, ): - parsed_args = get_command_arguments() - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_testcraft_yaml_j2 - ), - ) - init_command.run(parsed_args) + """Initialize a project with a single template file.""" + init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") with check: assert emitter.assert_message("Successfully initialised project.") @@ -217,43 +186,34 @@ def test_init_works_with_single_template_file( assert project.name == fake_empty_project_dir.name +@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_works_with_single_template_and_custom_name( - init_command: InitCommand, - fake_empty_project_dir: Path, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, project_yaml_filename: str, - template_dir_with_testcraft_yaml_j2: Path, emitter, - mocker, check, ): - custom_name = "some-other-testproject" - parsed_args = get_command_arguments(name=custom_name) - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_testcraft_yaml_j2 - ), - ) - init_command.run(parsed_args) + """Initialize a project with a single template file and custom name.""" + name = "some-other-test-project" + init_service.run(project_dir=fake_empty_project_dir, name=name) with check: assert emitter.assert_message("Successfully initialised project.") - project_yaml_path = Path(fake_empty_project_dir, project_yaml_filename) + project_yaml_path = pathlib.Path(fake_empty_project_dir, project_yaml_filename) with check: assert project_yaml_path.exists(), "Project should be initialized with template" project = Project.from_yaml_file(project_yaml_path) with check: - assert project.name == custom_name + assert project.name == name def check_file_existence_and_content( - check, - file_path: Path, - content: str, + check, file_path: pathlib.Path, content: str ) -> None: + """Helper function to ensure a file exists and has the correct content.""" with check: assert file_path.exists(), f"{file_path.name} should be created" @@ -261,30 +221,14 @@ def check_file_existence_and_content( assert file_path.read_text() == content, f"{file_path.name} incorrect content" +@pytest.mark.usefixtures("template_dir_with_multiple_non_ninja_files") def test_init_works_with_non_jinja2_templates( - init_command: InitCommand, - fake_empty_project_dir: Path, - template_dir_with_multiple_non_ninja_files: Path, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, emitter, - mocker, check, ): - parsed_args = get_command_arguments() - - mocker.patch.object( - init_command, - "_get_template_dir", - return_value=template_dir_with_multiple_non_ninja_files, - ) - - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_multiple_non_ninja_files - ), - ) - init_command.run(parsed_args) + init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") with check: assert emitter.assert_message("Successfully initialised project.") @@ -299,28 +243,14 @@ def test_init_works_with_non_jinja2_templates( ) +@pytest.mark.usefixtures("template_dir_with_symlinks") def test_init_does_not_follow_symlinks_but_copies_them_as_is( - init_command: InitCommand, - fake_empty_project_dir: Path, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, project_yaml_filename: str, - template_dir_with_symlinks: Path, - mocker, check, ): - parsed_args = get_command_arguments() - - mocker.patch.object( - init_command, - "_get_template_dir", - return_value=template_dir_with_symlinks, - ) - - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment(template_dir_with_symlinks), - ) - init_command.run(parsed_args) + init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") project = Project.from_yaml_file(fake_empty_project_dir / project_yaml_filename) with check: @@ -331,32 +261,18 @@ def test_init_does_not_follow_symlinks_but_copies_them_as_is( ).is_symlink(), "Symlink should be left intact." +@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_does_not_fail_on_non_empty_dir( - init_command: InitCommand, - non_empty_project_dir: Path, + init_service: InitService, + non_empty_project_dir: pathlib.Path, project_yaml_filename: str, - template_dir_with_testcraft_yaml_j2: Path, - mocker, emitter, check, ): - parsed_args = get_command_arguments() - - mocker.patch.object( - init_command, - "_get_template_dir", - return_value=template_dir_with_testcraft_yaml_j2, + init_service.run( + project_dir=non_empty_project_dir, name="fake-non-empty-project-dir" ) - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_testcraft_yaml_j2 - ), - ) - init_command.run(parsed_args) - with check: assert emitter.assert_message("Successfully initialised project.") @@ -369,33 +285,18 @@ def test_init_does_not_fail_on_non_empty_dir( assert project.name == non_empty_project_dir.name +@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_does_not_override_existing_craft_yaml( - init_command: InitCommand, - non_empty_project_dir: Path, + init_service: InitService, + non_empty_project_dir: pathlib.Path, project_yaml_filename: str, fake_project: Project, - template_dir_with_testcraft_yaml_j2: Path, - mocker, emitter, check, ): - parsed_args = get_command_arguments(project_dir=non_empty_project_dir) fake_project.to_yaml_file(non_empty_project_dir / project_yaml_filename) - mocker.patch.object( - init_command, - "_get_template_dir", - return_value=template_dir_with_testcraft_yaml_j2, - ) - - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_testcraft_yaml_j2 - ), - ) - init_command.run(parsed_args) + init_service.run(project_dir=non_empty_project_dir, name="fake-project-dir") with check: assert emitter.assert_message("Successfully initialised project.") @@ -409,35 +310,17 @@ def test_init_does_not_override_existing_craft_yaml( assert project.name == fake_project.name +@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_with_different_name_and_directory( - init_command: InitCommand, - fake_empty_project_dir: Path, - template_dir_with_testcraft_yaml_j2: Path, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, project_yaml_filename: str, - mocker, emitter, check, ): - custom_project_name = "some-custom-project" - parsed_args = get_command_arguments( - name=custom_project_name, - project_dir=fake_empty_project_dir, - ) + name = "some-custom-project" - mocker.patch.object( - init_command, - "_get_template_dir", - return_value=template_dir_with_testcraft_yaml_j2, - ) - - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_testcraft_yaml_j2 - ), - ) - init_command.run(parsed_args) + init_service.run(project_dir=fake_empty_project_dir, name=name) with check: assert emitter.assert_message("Successfully initialised project.") @@ -448,36 +331,20 @@ def test_init_with_different_name_and_directory( assert project_yaml_path.exists(), "Project should be initialized with template" project = Project.from_yaml_file(project_yaml_path) with check: - assert project.name == custom_project_name + assert project.name == name +@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_with_default_arguments_uses_current_directory( - init_command: InitCommand, - fake_empty_project_dir: Path, - template_dir_with_testcraft_yaml_j2: Path, + init_service: InitService, + fake_empty_project_dir: pathlib.Path, project_yaml_filename: str, - mocker, emitter, check, ): expected_project_name = fake_empty_project_dir.name - parsed_args = get_command_arguments() - - mocker.patch.object( - init_command, - "_get_template_dir", - return_value=template_dir_with_testcraft_yaml_j2, - ) - - mocker.patch.object( - init_command, - "_get_templates_environment", - return_value=get_jinja2_template_environment( - template_dir_with_testcraft_yaml_j2 - ), - ) - init_command.run(parsed_args) + init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") with check: assert emitter.assert_message("Successfully initialised project.") diff --git a/tests/integration/test_application.py b/tests/integration/test_application.py index 536f660c..d758c26c 100644 --- a/tests/integration/test_application.py +++ b/tests/integration/test_application.py @@ -72,11 +72,12 @@ def app(create_app): -V, --version: Show the application version and exit Starter commands: + init: Create an initial project filetree version: Show the application version and exit Commands can be classified as follows: Lifecycle: build, clean, pack, prime, pull, stage - Other: version + Other: init, version For more information about a command, run 'testcraft help '. For a summary of all commands, run 'testcraft help --all'. diff --git a/tests/unit/commands/test_init.py b/tests/unit/commands/test_init.py new file mode 100644 index 00000000..5f0019ee --- /dev/null +++ b/tests/unit/commands/test_init.py @@ -0,0 +1,58 @@ +# This file is part of craft-application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . + +"""Tests for init command.""" + +import argparse +import pathlib + +import pytest +from craft_application.commands import InitCommand + +# init operates in the current working directory +pytestmark = pytest.mark.usefixtures("new_dir") + + +@pytest.fixture +def init_command(app_metadata, mock_services): + return InitCommand({"app": app_metadata, "services": mock_services}) + + +@pytest.mark.parametrize("name", [None, "my-project"]) +def test_init_in_cwd(init_command, name, new_dir, mock_services): + """Test the init command in the current working directory.""" + expected_name = name or new_dir.name + parsed_args = argparse.Namespace(project_dir=None, name=name) + + init_command.run(parsed_args) + + mock_services.init.run.assert_called_once_with( + project_dir=new_dir, name=expected_name + ) + + +@pytest.mark.parametrize("name", [None, "my-project"]) +def test_init_run_project_dir(init_command, name, mock_services): + """Test the init command in a project directly.""" + expected_name = name or "test-project-dir" + project_dir = pathlib.Path("test-project-dir") + parsed_args = argparse.Namespace(project_dir=project_dir, name=name) + + init_command.run(parsed_args) + + mock_services.init.run.assert_called_once_with( + project_dir=project_dir, name=expected_name + ) diff --git a/tests/unit/commands/test_other.py b/tests/unit/commands/test_other.py index fcfef2b5..95a40b5c 100644 --- a/tests/unit/commands/test_other.py +++ b/tests/unit/commands/test_other.py @@ -13,15 +13,14 @@ # # You should have received a copy of the GNU Lesser General Public License along # with this program. If not, see . -"""Tests for lifecycle commands.""" +"""Tests for other commands.""" import argparse import pytest +from craft_application.commands import InitCommand from craft_application.commands.other import VersionCommand, get_other_command_group -OTHER_COMMANDS = { - VersionCommand, -} +OTHER_COMMANDS = {InitCommand, VersionCommand} @pytest.mark.parametrize("commands", [OTHER_COMMANDS]) diff --git a/tests/unit/conftest.py b/tests/unit/conftest.py index 33749dd3..7b9b5c98 100644 --- a/tests/unit/conftest.py +++ b/tests/unit/conftest.py @@ -51,4 +51,5 @@ def mock_services(app_metadata, fake_project, fake_package_service_class): factory.provider = mock.Mock(spec=services.ProviderService) factory.remote_build = mock.Mock(spec_set=services.RemoteBuildService) factory.fetch = mock.Mock(spec=services.FetchService) + factory.init = mock.Mock(spec=services.InitService) return factory diff --git a/tests/unit/remote/conftest.py b/tests/unit/remote/conftest.py deleted file mode 100644 index f64a3ec5..00000000 --- a/tests/unit/remote/conftest.py +++ /dev/null @@ -1,29 +0,0 @@ -# Copyright (C) 2022,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 . -import os -import pathlib - -import pytest - - -@pytest.fixture -def new_dir(tmp_path): - """Change to a new temporary directory.""" - - cwd = pathlib.Path.cwd() - os.chdir(tmp_path) - - yield tmp_path - - os.chdir(cwd) diff --git a/tests/unit/services/test_init.py b/tests/unit/services/test_init.py new file mode 100644 index 00000000..5c640e84 --- /dev/null +++ b/tests/unit/services/test_init.py @@ -0,0 +1,205 @@ +# This file is part of craft-application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . + +"""Unit tests for the InitService.""" + +import pathlib +from unittest import mock + +import jinja2 +import pytest +import pytest_check +from craft_application import services + + +@pytest.fixture +def init_service(app_metadata, fake_services): + _init_service = services.InitService(app_metadata, fake_services) + _init_service.setup() + return _init_service + + +@pytest.fixture +def mock_template(): + _mock_template = mock.Mock(spec=jinja2.Template) + _mock_template.render.return_value = "rendered content" + return _mock_template + + +@pytest.fixture +def mock_environment(mock_template): + _mock_environment = mock.Mock(spec=jinja2.Environment) + _mock_environment.get_template.return_value = mock_template + return _mock_environment + + +def test_get_context(init_service): + context = init_service._get_context(name="my-project") + + assert context == {"name": "my-project"} + + +def test_get_template_dir(init_service): + template_dir = init_service._get_template_dir() + + assert template_dir == pathlib.Path("templates") + + +def test_get_executable_files(init_service): + executable_files = init_service._get_executable_files() + + assert executable_files == [] + + +def test_create_project_dir(init_service, tmp_path, emitter): + project_dir = tmp_path / "my-project" + + init_service._create_project_dir(project_dir=project_dir, name="my-project") + + assert project_dir.is_dir() + emitter.assert_debug(f"Using project directory {str(project_dir)!r} for my-project") + + +def test_create_project_dir_exists(init_service, tmp_path, emitter): + """Do not error if the project directory already exists.""" + project_dir = tmp_path / "my-project" + project_dir.mkdir() + + init_service._create_project_dir(project_dir=project_dir, name="my-project") + + assert project_dir.is_dir() + emitter.assert_debug(f"Using project directory {str(project_dir)!r} for my-project") + + +def test_get_templates_environment(init_service, mocker): + """Test that _get_templates_environment returns a Jinja2 environment.""" + mock_package_loader = mocker.patch("jinja2.PackageLoader") + mock_environment = mocker.patch("jinja2.Environment") + + environment = init_service._get_templates_environment(pathlib.Path("test-dir")) + + mock_package_loader.assert_called_once_with("testcraft", "test-dir") + mock_environment.assert_called_once_with( + loader=mock_package_loader.return_value, + autoescape=False, + keep_trailing_newline=True, + optimized=False, + undefined=jinja2.StrictUndefined, + ) + assert environment == mock_environment.return_value + + +@pytest.mark.parametrize("template_filename", ["file1.txt", "nested/file2.txt"]) +def test_copy_template_file(init_service, tmp_path, template_filename): + # create template + template_dir = tmp_path / "templates" + template_file = template_dir / template_filename + template_file.parent.mkdir(parents=True, exist_ok=True) + template_file.write_text("content") + # create project with an existing file + project_dir = tmp_path / "project" + project_dir.mkdir() + + init_service._copy_template_file(template_filename, template_dir, project_dir) + + assert (project_dir / template_filename).read_text() == "content" + + +@pytest.mark.parametrize("template_name", ["file1.txt", "nested/file2.txt"]) +def test_copy_template_file_exists(init_service, tmp_path, template_name, emitter): + """Do not overwrite existing files.""" + # create template + template_dir = tmp_path / "templates" + template_file = template_dir / template_name + template_file.parent.mkdir(parents=True, exist_ok=True) + template_file.write_text("content") + # create project with an existing file + project_dir = tmp_path / "project" + (project_dir / template_name).parent.mkdir(parents=True, exist_ok=True) + (project_dir / template_name).write_text("existing content") + + init_service._copy_template_file(template_name, template_dir, project_dir) + + assert (project_dir / template_name).read_text() == "existing content" + emitter.assert_debug( + f"Skipping file {template_name} because it is already present." + ) + + +def test_render_project(init_service, tmp_path, mock_environment): + template_filenames = [ + # Jinja2 templates + "jinja-file.txt.j2", + "nested/jinja-file.txt.j2", + # Non-Jinja2 templates (regular files) + "non-jinja-file.txt", + "nested/non-jinja-file.txt", + ] + mock_environment.list_templates.return_value = template_filenames + project_dir = tmp_path / "project" + project_dir.mkdir() + template_dir = tmp_path / "templates" + for filename in template_filenames: + (template_dir / filename).parent.mkdir(parents=True, exist_ok=True) + (template_dir / filename).write_text("template content") + + init_service._render_project( + environment=mock_environment, + project_dir=project_dir, + template_dir=template_dir, + context={"name": "my-project"}, + executable_files=[], + ) + + pytest_check.equal((project_dir / "jinja-file.txt").read_text(), "rendered content") + pytest_check.equal( + (project_dir / "nested/jinja-file.txt").read_text(), "rendered content" + ) + pytest_check.equal( + (project_dir / "non-jinja-file.txt").read_text(), "template content" + ) + pytest_check.equal( + (project_dir / "nested/non-jinja-file.txt").read_text(), "template content" + ) + + +def test_render_project_executable(init_service, tmp_path, mock_environment): + template_filenames = [ + "executable-1.sh.j2", + "executable-2.sh.j2", + "executable-3.sh", + ] + mock_environment.list_templates.return_value = template_filenames + project_dir = tmp_path / "project" + project_dir.mkdir() + template_dir = tmp_path / "templates" + for filename in template_filenames: + (template_dir / filename).parent.mkdir(parents=True, exist_ok=True) + (template_dir / filename).write_text("template content") + + init_service._render_project( + environment=mock_environment, + project_dir=project_dir, + template_dir=template_dir, + context={"name": "my-project"}, + executable_files=["executable-1.sh", "executable-3.sh"], + ) + + import os + + pytest_check.is_true(os.access(project_dir / "executable-1.sh", os.X_OK)) + pytest_check.is_false(os.access(project_dir / "executable-2.sh", os.X_OK)) + pytest_check.is_false(os.access(project_dir / "executable-3.sh", os.X_OK)) diff --git a/tests/unit/util/test_file.py b/tests/unit/util/test_file.py new file mode 100644 index 00000000..1c6c860c --- /dev/null +++ b/tests/unit/util/test_file.py @@ -0,0 +1,54 @@ +# This file is part of craft_application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . + +"""Unit tests for utility functions and helpers related to path handling.""" + +from _stat import S_IRGRP, S_IROTH, S_IRUSR, S_IWUSR, S_IXGRP, S_IXOTH, S_IXUSR + +import pytest +from craft_application.util import file + +USER = S_IRUSR | S_IWUSR # 0o600 +USER_GROUP = USER | S_IRGRP # 0o640 +USER_GROUP_OTHER = USER_GROUP | S_IROTH # 0o644 + + +@pytest.mark.parametrize( + ("initial_permissions", "expected_permissions"), + [ + pytest.param(USER, USER | S_IXUSR, id="user"), + pytest.param( + USER_GROUP, + USER_GROUP | S_IXUSR | S_IXGRP, + id="user-group", + ), + pytest.param( + USER_GROUP_OTHER, + USER_GROUP_OTHER | S_IXUSR | S_IXGRP | S_IXOTH, + id="user-group-other", + ), + ], +) +def test_make_executable_read_bits(initial_permissions, expected_permissions, tmp_path): + """make_executable should only operate where the read bits are set.""" + test_file = tmp_path / "test" + test_file.touch(mode=initial_permissions) + + with test_file.open() as fd: + file.make_executable(fd) + + # only read bits got made executable + assert test_file.stat().st_mode & 0o777 == expected_permissions From 2e8007b1030db86c162acb5da066125ac962730e Mon Sep 17 00:00:00 2001 From: Callahan Kovacs Date: Thu, 17 Oct 2024 11:13:19 -0500 Subject: [PATCH 3/4] feat(init): add profile support Signed-off-by: Callahan Kovacs --- craft_application/commands/init.py | 73 +++++++- craft_application/errors.py | 4 + craft_application/services/init.py | 139 ++++++++++---- craft_application/util/__init__.py | 3 +- craft_application/util/file.py | 12 ++ docs/reference/changelog.rst | 18 ++ tests/conftest.py | 18 +- tests/integration/commands/__init__.py | 0 tests/integration/commands/test_init.py | 158 ++++++++++++++++ tests/integration/services/test_init.py | 235 ++++++++++++++++++------ tests/unit/commands/test_init.py | 77 +++++++- tests/unit/services/test_init.py | 170 ++++++++++------- tests/unit/util/test_file.py | 18 +- 13 files changed, 739 insertions(+), 186 deletions(-) create mode 100644 tests/integration/commands/__init__.py create mode 100644 tests/integration/commands/test_init.py diff --git a/craft_application/commands/init.py b/craft_application/commands/init.py index d1bb7324..8395241f 100644 --- a/craft_application/commands/init.py +++ b/craft_application/commands/init.py @@ -16,15 +16,23 @@ from __future__ import annotations import argparse +import importlib.resources import pathlib from textwrap import dedent from typing import cast +import craft_cli + +from craft_application.util import humanize_list + from . import base class InitCommand(base.AppCommand): - """Command to create initial project files.""" + """Command to create initial project files. + + The init command should always produce a working and ready-to-build project. + """ name = "init" help_msg = "Create an initial project filetree" @@ -32,12 +40,23 @@ class InitCommand(base.AppCommand): """ Initialise a project. - If is provided, initialise in that directory, + If '' is provided, initialise in that directory, otherwise initialise in the current working directory. + + If '--name ' is provided, the project will be named ''. + Otherwise, the project will be named after the directory it is initialised in. + + '--profile ' is used to initialise the project for a specific use case. + + Init can work in an existing project directory. If there are any files in the + directory that would be overwritten, then init command will fail. """ ) common = True + default_profile = "simple" + """The default profile to use when initialising a project.""" + def fill_parser(self, parser: argparse.ArgumentParser) -> None: """Specify command's specific parameters.""" parser.add_argument( @@ -45,7 +64,7 @@ def fill_parser(self, parser: argparse.ArgumentParser) -> None: type=pathlib.Path, nargs="?", default=None, - help="Path to initialize project in; defaults to current working directory.", + help="Path to initialise project in; defaults to current working directory.", ) parser.add_argument( "--name", @@ -53,15 +72,55 @@ def fill_parser(self, parser: argparse.ArgumentParser) -> None: default=None, help="The name of project; defaults to the name of ", ) + # TODO: this fails to render in `--help` (#530) + parser.add_argument( + "--profile", + type=str, + choices=self.profiles, + default=self.default_profile, + help=( + f"Use the specified project profile (default is {self.default_profile}, " + f"choices are {humanize_list(self.profiles, 'and')})" + ), + ) + + @property + def parent_template_dir(self) -> pathlib.Path: + """Return the path to the directory that contains all templates.""" + with importlib.resources.path( + self._app.name, "templates" + ) as _parent_template_dir: + return _parent_template_dir + + @property + def profiles(self) -> list[str]: + """A list of profile names generated from template directories.""" + template_dirs = [ + path for path in self.parent_template_dir.iterdir() if path.is_dir() + ] + return sorted([template.name for template in template_dirs]) def run(self, parsed_args: argparse.Namespace) -> None: """Run the command.""" project_dir = self._get_project_dir(parsed_args) - name = self._get_name(parsed_args) - self._services.init.run(project_dir=project_dir, name=name) + project_name = self._get_name(parsed_args) + template_dir = pathlib.Path(self.parent_template_dir / parsed_args.profile) + + craft_cli.emit.progress("Checking project directory.") + self._services.init.check_for_existing_files( + project_dir=project_dir, template_dir=template_dir + ) + + craft_cli.emit.progress("Initialising project.") + self._services.init.initialise_project( + project_dir=project_dir, + project_name=project_name, + template_dir=template_dir, + ) + craft_cli.emit.message("Successfully initialised project.") def _get_name(self, parsed_args: argparse.Namespace) -> str: - """Get name of the package that is about to be initialized. + """Get name of the package that is about to be initialised. Check if name is set explicitly or fallback to project_dir. """ @@ -71,7 +130,7 @@ def _get_name(self, parsed_args: argparse.Namespace) -> str: @staticmethod def _get_project_dir(parsed_args: argparse.Namespace) -> pathlib.Path: - """Get project dir where project should be initialized. + """Get project dir where project should be initialised. It applies rules in the following order: - if is specified explicitly, it returns diff --git a/craft_application/errors.py b/craft_application/errors.py index 0c4a74af..a0635730 100644 --- a/craft_application/errors.py +++ b/craft_application/errors.py @@ -246,3 +246,7 @@ def __init__( # (too many arguments) class FetchServiceError(CraftError): """Errors related to the fetch-service.""" + + +class InitError(CraftError): + """Errors related to initialising a project.""" diff --git a/craft_application/services/init.py b/craft_application/services/init.py index 71b016e4..99b55b59 100644 --- a/craft_application/services/init.py +++ b/craft_application/services/init.py @@ -24,7 +24,8 @@ import jinja2 from craft_cli import emit -from craft_application.util import make_executable +from craft_application.errors import InitError +from craft_application.util import is_executable, make_executable from . import base @@ -32,21 +33,65 @@ class InitService(base.AppService): """Service class for initializing a project.""" - def run(self, *, project_dir: pathlib.Path, name: str) -> None: - """Initialize a new project.""" - context = self._get_context(name=name) - template_dir = self._get_template_dir() - environment = self._get_templates_environment(template_dir) - executable_files = self._get_executable_files() - - self._create_project_dir(project_dir=project_dir, name=name) - self._render_project( - environment, - project_dir, - template_dir, - context, - executable_files, + def initialise_project( + self, + *, + project_dir: pathlib.Path, + project_name: str, + template_dir: pathlib.Path, + ) -> None: + """Initialise a new project from a template. + + If a file already exists in the project directory, it is not overwritten. + Use `check_for_existing_files()` to see if this will occur before initialising + the project. + + :param project_dir: The directory to initialise the project in. + :param project_name: The name of the project. + :param template_dir: The directory containing the templates. + """ + emit.debug( + f"Initialising project {project_name!r} in {str(project_dir)!r} from " + f"template in {str(template_dir)!r}." ) + context = self._get_context(name=project_name) + environment = self._get_templates_environment(template_dir) + self._create_project_dir(project_dir=project_dir) + self._render_project(environment, project_dir, template_dir, context) + + def check_for_existing_files( + self, + *, + project_dir: pathlib.Path, + template_dir: pathlib.Path, + ) -> None: + """Check if there are any existing files in the project directory that would be overwritten. + + :param project_dir: The directory to initialise the project in. + :param template_dir: The directory containing the templates. + + :raises InitError: If there are files in the project directory that would be overwritten. + """ + template_files = self._get_template_files(template_dir) + existing_files = [ + template_file + for template_file in template_files + if (project_dir / template_file).exists() + ] + + if existing_files: + existing_files_formatted = "\n - ".join(existing_files) + raise InitError( + message=( + f"Cannot initialise project in {str(project_dir)!r} because it " + "would overwrite existing files.\nExisting files are:\n - " + f"{existing_files_formatted}" + ), + resolution=( + "Initialise the project in an empty directory or remove the existing files." + ), + retcode=os.EX_CANTCREAT, + ) def _copy_template_file( self, @@ -77,7 +122,6 @@ def _render_project( project_dir: pathlib.Path, template_dir: pathlib.Path, context: dict[str, Any], - executable_files: list[str], ) -> None: """Render files for a project from a template. @@ -85,8 +129,8 @@ def _render_project( :param project_dir: The directory to render the files into. :param template_dir: The directory where templates are stored. :param context: The context to render the templates with. - :param executable_files: The list of files that should be executable. """ + emit.progress("Rendering project.") for template_name in environment.list_templates(): if not template_name.endswith(".j2"): self._copy_template_file(template_name, template_dir, project_dir) @@ -105,18 +149,10 @@ def _render_project( with path.open("wt", encoding="utf8") as file: out = template.render(context) file.write(out) - if rendered_template_name in executable_files and os.name == "posix": + if is_executable(template_dir / template_name) and os.name == "posix": make_executable(file) emit.debug(" made executable") - emit.message("Successfully initialised project.") - - def _get_template_dir(self) -> pathlib.Path: - """Return the path to the template directory.""" - return pathlib.Path("templates") - - def _get_executable_files(self) -> list[str]: - """Return the list of files that should be executable.""" - return [] + emit.progress("Rendered project.") def _get_context(self, name: str) -> dict[str, Any]: """Get context to render templates with. @@ -127,25 +163,36 @@ def _get_context(self, name: str) -> dict[str, Any]: return {"name": name} - def _create_project_dir( - self, - project_dir: pathlib.Path, - *, - name: str, - ) -> None: - """Create the project path if it does not already exist.""" + @staticmethod + def _create_project_dir(project_dir: pathlib.Path) -> None: + """Create the project directory if it does not already exist.""" if not project_dir.exists(): + emit.debug(f"Creating project directory {str(project_dir)!r}.") project_dir.mkdir(parents=True) - emit.debug(f"Using project directory {str(project_dir)!r} for {name}") + else: + emit.debug( + f"Not creating project directory {str(project_dir)!r} " + "because it already exists." + ) + + def _get_loader(self, template_dir: pathlib.Path) -> jinja2.BaseLoader: + """Return a Jinja loader for the given template directory. - def _get_loader(self, template_dir: pathlib.Path) -> jinja2.PackageLoader: - """Return a Jinja template loader for the given template directory.""" + :param template_dir: The directory containing the templates. + + :returns: A Jinja loader. + """ return jinja2.PackageLoader(self._app.name, str(template_dir)) def _get_templates_environment( self, template_dir: pathlib.Path ) -> jinja2.Environment: - """Create and return a Jinja environment to deal with the templates.""" + """Create and return a Jinja environment to deal with the templates. + + :param template_dir: The directory containing the templates. + + :returns: A Jinja environment. + """ return jinja2.Environment( loader=self._get_loader(template_dir), autoescape=False, # noqa: S701 (jinja2-autoescape-false) @@ -153,3 +200,19 @@ def _get_templates_environment( optimized=False, # optimization doesn't make sense for one-offs undefined=jinja2.StrictUndefined, ) # fail on undefined + + def _get_template_files(self, template_dir: pathlib.Path) -> list[str]: + """Return a list of files that would be created from a template directory. + + Note that the '.j2' suffix is removed from templates. + + :param template_dir: The directory containing the templates. + + :returns: A list of filenames that would be created. + """ + templates = self._get_templates_environment(template_dir).list_templates() + + return [ + template[:-3] if template.endswith(".j2") else template + for template in templates + ] diff --git a/craft_application/util/__init__.py b/craft_application/util/__init__.py index 49e913e2..0498cfb9 100644 --- a/craft_application/util/__init__.py +++ b/craft_application/util/__init__.py @@ -17,7 +17,7 @@ from craft_application.util.callbacks import get_unique_callbacks from craft_application.util.docs import render_doc_url -from craft_application.util.file import make_executable +from craft_application.util.file import is_executable, make_executable from craft_application.util.logging import setup_loggers from craft_application.util.paths import get_filename_from_url_path, get_managed_logpath from craft_application.util.platforms import ( @@ -39,6 +39,7 @@ __all__ = [ "get_unique_callbacks", "render_doc_url", + "is_executable", "make_executable", "setup_loggers", "get_filename_from_url_path", diff --git a/craft_application/util/file.py b/craft_application/util/file.py index 7a0e4e62..aa1a96dd 100644 --- a/craft_application/util/file.py +++ b/craft_application/util/file.py @@ -18,16 +18,28 @@ import io import os +import pathlib from _stat import S_IRGRP, S_IROTH, S_IRUSR S_IRALL = S_IRUSR | S_IRGRP | S_IROTH """0o444 permission mask for execution permissions for everybody.""" +def is_executable(filepath: pathlib.Path) -> bool: + """Check if file is executable. + + :param filepath: The file to check. + + :returns: True if file exists and is executable. + """ + return os.access(filepath, os.X_OK) + + def make_executable(fh: io.IOBase) -> None: """Make open file fh executable. Only makes the file executable for the user, group, or other if they already had read permissions. + Effectively, this makes the file executable with the same umask as when the file was created. :param fh: An open file object. """ diff --git a/docs/reference/changelog.rst b/docs/reference/changelog.rst index b8447809..0e5ad8ac 100644 --- a/docs/reference/changelog.rst +++ b/docs/reference/changelog.rst @@ -4,6 +4,24 @@ Changelog ********* +X.Y.Z (2024-MMM-DD) +------------------- + +Commands +======== + +- Adds an ``init`` command for initialising new projects. + + +Services +======== + +- Adds an ``InitService`` for initialising new projects. + +.. + For a complete list of commits, check out the `X.Y.Z`_ release on GitHub. + + 4.3.0 (2024-Oct-11) ------------------- diff --git a/tests/conftest.py b/tests/conftest.py index c198584f..ac105b72 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -24,11 +24,13 @@ import craft_application import craft_parts +import jinja2 import pydantic import pytest from craft_application import application, launchpad, models, services, util from craft_cli import EmitterMode, emit from craft_providers import bases +from jinja2 import FileSystemLoader from typing_extensions import override if TYPE_CHECKING: # pragma: no cover @@ -295,15 +297,29 @@ def __init__( return FakeLifecycleService +@pytest.fixture +def fake_init_service_class(tmp_path): + class FakeInitService(services.InitService): + def _get_loader(self, template_dir: pathlib.Path) -> jinja2.BaseLoader: + return FileSystemLoader(tmp_path / "templates" / template_dir) + + return FakeInitService + + @pytest.fixture def fake_services( - app_metadata, fake_project, fake_lifecycle_service_class, fake_package_service_class + app_metadata, + fake_project, + fake_lifecycle_service_class, + fake_package_service_class, + fake_init_service_class, ): return services.ServiceFactory( app_metadata, project=fake_project, PackageClass=fake_package_service_class, LifecycleClass=fake_lifecycle_service_class, + InitClass=fake_init_service_class, ) diff --git a/tests/integration/commands/__init__.py b/tests/integration/commands/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/tests/integration/commands/test_init.py b/tests/integration/commands/test_init.py new file mode 100644 index 00000000..13cab7ba --- /dev/null +++ b/tests/integration/commands/test_init.py @@ -0,0 +1,158 @@ +# This file is part of craft-application. +# +# Copyright 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 warranties of MERCHANTABILITY, +# SATISFACTORY QUALITY, 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 . + +"""Tests for init command.""" +import os +import pathlib +import textwrap + +import pytest +from craft_application.commands import InitCommand + +# init operates in the current working directory +pytestmark = pytest.mark.usefixtures("new_dir") + + +@pytest.fixture(autouse=True) +def mock_parent_template_dir(tmp_path, mocker): + """Mock the parent template directory.""" + mocker.patch.object( + InitCommand, + "parent_template_dir", + pathlib.Path(tmp_path) / "templates", + ) + + +@pytest.fixture +def fake_template_dirs(tmp_path): + """Set up a fake template directories with two templates. + + These templates are very simple because the InitService tests focus on the + templates themselves. + """ + parent_template_dir = tmp_path / "templates" + + simple_template_file = parent_template_dir / "simple" / "simple-file.j2" + simple_template_file.parent.mkdir(parents=True) + simple_template_file.write_text("name={{ name }}") + + other_template_file = parent_template_dir / "other-template" / "other-file.j2" + other_template_file.parent.mkdir(parents=True) + other_template_file.write_text("name={{ name }}") + + +@pytest.mark.parametrize( + ("profile", "expected_file"), + [ + (None, pathlib.Path("simple-file")), + ("simple", pathlib.Path("simple-file")), + ("other-template", pathlib.Path("other-file")), + ], +) +@pytest.mark.parametrize("project_dir", [None, "project-dir"]) +@pytest.mark.usefixtures("fake_template_dirs") +def test_init(app, capsys, monkeypatch, profile, expected_file, project_dir): + """Initialise a project.""" + expected_output = "Successfully initialised project" + command = ["testcraft", "init"] + if profile: + command.extend(["--profile", profile]) + if project_dir: + command.append(project_dir) + expected_file = pathlib.Path(project_dir) / expected_file + monkeypatch.setattr("sys.argv", command) + + return_code = app.run() + stdout, _ = capsys.readouterr() + + assert return_code == os.EX_OK + assert expected_output in stdout + assert expected_file.is_file() + # name is not provided, so use the project directory name + assert f"name={expected_file.resolve().parent.name}" == expected_file.read_text() + + +@pytest.mark.usefixtures("fake_template_dirs") +@pytest.mark.parametrize( + ("project_dir", "expected_file"), + [ + (None, pathlib.Path("simple-file")), + ("project-dir", pathlib.Path("project-dir") / "simple-file"), + ], +) +def test_init_name(app, capsys, monkeypatch, project_dir, expected_file): + """Initialise a project with a name.""" + expected_output = "Successfully initialised project" + command = ["testcraft", "init", "--name", "test-project-name"] + if project_dir: + command.append(project_dir) + monkeypatch.setattr("sys.argv", command) + + return_code = app.run() + stdout, _ = capsys.readouterr() + + assert return_code == os.EX_OK + assert expected_output in stdout + assert expected_file.is_file() + assert expected_file.read_text() == "name=test-project-name" + + +@pytest.mark.usefixtures("fake_template_dirs") +def test_init_invalid_profile(app, capsys, monkeypatch): + """Give a helpful error message for invalid profiles.""" + expected_error = "Error: argument --profile: invalid choice: 'bad' (choose from 'other-template', 'simple')" + monkeypatch.setattr("sys.argv", ["testcraft", "init", "--profile", "bad"]) + + return_code = app.run() + _, stderr = capsys.readouterr() + + assert return_code == os.EX_USAGE + assert expected_error in stderr + + +@pytest.mark.usefixtures("fake_template_dirs") +def test_init_overlapping_file(app, capsys, monkeypatch, tmp_path): + """Give a helpful error message if a file would be overwritten.""" + pathlib.Path("simple-file").touch() + expected_error = textwrap.dedent( + f""" + Cannot initialise project in {str(tmp_path)!r} because it would overwrite existing files. + Existing files are: + - simple-file + Recommended resolution: Initialise the project in an empty directory or remove the existing files.""" + ) + monkeypatch.setattr("sys.argv", ["testcraft", "init", "--profile", "simple"]) + + return_code = app.run() + _, stderr = capsys.readouterr() + + assert return_code == os.EX_CANTCREAT + assert expected_error in stderr + + +@pytest.mark.usefixtures("fake_template_dirs") +def test_init_nonoverlapping_file(app, capsys, monkeypatch): + """Files can exist in the project directory if they won't be overwritten.""" + expected_output = "Successfully initialised project" + pathlib.Path("unrelated-file").touch() + monkeypatch.setattr("sys.argv", ["testcraft", "init", "--profile", "simple"]) + + return_code = app.run() + stdout, _ = capsys.readouterr() + + assert return_code == os.EX_OK + assert expected_output in stdout + assert pathlib.Path("simple-file").is_file() diff --git a/tests/integration/services/test_init.py b/tests/integration/services/test_init.py index dcad9838..a75b5a7b 100644 --- a/tests/integration/services/test_init.py +++ b/tests/integration/services/test_init.py @@ -17,29 +17,25 @@ """Tests for init service.""" import pathlib +import subprocess import sys import textwrap import pytest +from craft_application import errors from craft_application.models.project import Project from craft_application.services import InitService -from jinja2 import FileSystemLoader +from craft_application.util import make_executable # init operates in the current working directory pytestmark = pytest.mark.usefixtures("new_dir") @pytest.fixture -def init_service(app_metadata, fake_services, mocker, tmp_path): - _init_service = InitService(app_metadata, fake_services) +def init_service(fake_init_service_class, app_metadata, fake_services): + _init_service = fake_init_service_class(app_metadata, fake_services) _init_service.setup() - mocker.patch.object( - _init_service, - "_get_loader", - return_value=FileSystemLoader(tmp_path / "templates"), - ) - return _init_service @@ -55,17 +51,17 @@ def project_yaml_filename() -> str: return "testcraft.yaml" -# TODO: test nested templates - - @pytest.fixture def template_dir_with_testcraft_yaml_j2( fake_empty_template_dir: pathlib.Path, project_yaml_filename: str, ) -> pathlib.Path: - (fake_empty_template_dir / f"{project_yaml_filename}.j2").write_text( - textwrap.dedent( - """ + """Creates the same testcraft.yaml file in the top-level and nested directories. + + Normally a project would only have one testcraft.yaml file, but two are created for testing. + """ + template_text = textwrap.dedent( + """ # This file configures testcraft. # (Required) @@ -103,8 +99,12 @@ def template_dir_with_testcraft_yaml_j2( platforms: amd64: """ - ) ) + top_level_template = fake_empty_template_dir / f"{project_yaml_filename}.j2" + top_level_template.write_text(template_text) + nested_template = fake_empty_template_dir / "nested" / f"{project_yaml_filename}.j2" + nested_template.parent.mkdir() + nested_template.write_text(template_text) return fake_empty_template_dir @@ -130,6 +130,26 @@ def template_dir_with_symlinks( return template_dir_with_testcraft_yaml_j2 +@pytest.fixture +def template_dir_with_executables( + fake_empty_template_dir: pathlib.Path, +) -> pathlib.Path: + """Create executable templated and non-templated files.""" + for filename in [ + "file.sh", + "nested/file.sh", + "template.sh.j2", + "nested/template.sh.j2", + ]: + filepath = fake_empty_template_dir / filename + filepath.parent.mkdir(exist_ok=True) + with filepath.open("wt", encoding="utf8") as file: + file.write("#!/bin/bash\necho 'Hello, world!'") + make_executable(file) + + return fake_empty_template_dir + + @pytest.fixture def fake_empty_project_dir(tmp_path) -> pathlib.Path: empty_project_dir_path = pathlib.Path(tmp_path / "fake-project-dir") @@ -149,62 +169,80 @@ def non_empty_project_dir(tmp_path) -> pathlib.Path: def test_init_works_with_empty_templates_dir( init_service: InitService, fake_empty_project_dir: pathlib.Path, + fake_empty_template_dir: pathlib.Path, emitter, check, ): - """Initialize a project with an empty templates directory.""" - init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") + """Initialise a project with an empty templates directory.""" + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name="fake-project-dir", + template_dir=fake_empty_template_dir, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") with check: assert not list( fake_empty_project_dir.iterdir() - ), "Project dir should be initialized empty" + ), "Project dir should be initialised empty" -@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") -def test_init_works_with_single_template_file( +def test_init_works_with_simple_template( init_service: InitService, fake_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, project_yaml_filename: str, emitter, check, ): - """Initialize a project with a single template file.""" - init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") + """Initialise a project with a simple project template.""" + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name="fake-project-dir", + template_dir=template_dir_with_testcraft_yaml_j2, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") - project_yaml_path = fake_empty_project_dir / project_yaml_filename + project_yaml_paths = [ + fake_empty_project_dir / project_yaml_filename, + fake_empty_project_dir / "nested" / project_yaml_filename, + ] - with check: - assert project_yaml_path.exists(), "Project should be initialized with template" - project = Project.from_yaml_file(project_yaml_path) - with check: - assert project.name == fake_empty_project_dir.name + for project_yaml_path in project_yaml_paths: + with check: + assert ( + project_yaml_path.exists() + ), "Project should be initialised with template" + project = Project.from_yaml_file(project_yaml_path) + assert project.name == fake_empty_project_dir.name -@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_works_with_single_template_and_custom_name( init_service: InitService, fake_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, project_yaml_filename: str, emitter, check, ): - """Initialize a project with a single template file and custom name.""" + """Initialise a project with a single template file and custom name.""" name = "some-other-test-project" - init_service.run(project_dir=fake_empty_project_dir, name=name) + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name=name, + template_dir=template_dir_with_testcraft_yaml_j2, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") project_yaml_path = pathlib.Path(fake_empty_project_dir, project_yaml_filename) with check: - assert project_yaml_path.exists(), "Project should be initialized with template" + assert project_yaml_path.exists(), "Project should be initialised with template" project = Project.from_yaml_file(project_yaml_path) with check: assert project.name == name @@ -221,17 +259,21 @@ def check_file_existence_and_content( assert file_path.read_text() == content, f"{file_path.name} incorrect content" -@pytest.mark.usefixtures("template_dir_with_multiple_non_ninja_files") def test_init_works_with_non_jinja2_templates( init_service: InitService, fake_empty_project_dir: pathlib.Path, + template_dir_with_multiple_non_ninja_files: pathlib.Path, emitter, check, ): - init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name="fake-project-dir", + template_dir=template_dir_with_multiple_non_ninja_files, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") check_file_existence_and_content( check, fake_empty_project_dir / "file1.txt", "Content of file1.txt" @@ -243,14 +285,18 @@ def test_init_works_with_non_jinja2_templates( ) -@pytest.mark.usefixtures("template_dir_with_symlinks") def test_init_does_not_follow_symlinks_but_copies_them_as_is( init_service: InitService, fake_empty_project_dir: pathlib.Path, + template_dir_with_symlinks: pathlib.Path, project_yaml_filename: str, check, ): - init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name="fake-project-dir", + template_dir=template_dir_with_symlinks, + ) project = Project.from_yaml_file(fake_empty_project_dir / project_yaml_filename) with check: @@ -261,34 +307,59 @@ def test_init_does_not_follow_symlinks_but_copies_them_as_is( ).is_symlink(), "Symlink should be left intact." -@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") +def test_init_copies_executables( + init_service: InitService, + fake_empty_project_dir: pathlib.Path, + template_dir_with_executables: pathlib.Path, + check, +): + """Executability of template files should be preserved.""" + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name="fake-project-dir", + template_dir=template_dir_with_executables, + ) + + for filename in ["file.sh", "nested/file.sh", "template.sh", "nested/template.sh"]: + with check: + assert ( + subprocess.check_output( + [str(fake_empty_project_dir / filename)], text=True + ) + == "Hello, world!\n" + ) + + def test_init_does_not_fail_on_non_empty_dir( init_service: InitService, non_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, project_yaml_filename: str, emitter, check, ): - init_service.run( - project_dir=non_empty_project_dir, name="fake-non-empty-project-dir" + init_service.initialise_project( + project_dir=non_empty_project_dir, + project_name="fake-non-empty-project-dir", + template_dir=template_dir_with_testcraft_yaml_j2, ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") project_yaml_path = non_empty_project_dir / project_yaml_filename with check: - assert project_yaml_path.exists(), "Project should be initialized with template" + assert project_yaml_path.exists(), "Project should be initialised with template" project = Project.from_yaml_file(project_yaml_path) with check: assert project.name == non_empty_project_dir.name -@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_does_not_override_existing_craft_yaml( init_service: InitService, non_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2, project_yaml_filename: str, fake_project: Project, emitter, @@ -296,63 +367,111 @@ def test_init_does_not_override_existing_craft_yaml( ): fake_project.to_yaml_file(non_empty_project_dir / project_yaml_filename) - init_service.run(project_dir=non_empty_project_dir, name="fake-project-dir") + init_service.initialise_project( + project_dir=non_empty_project_dir, + project_name="fake-project-dir", + template_dir=template_dir_with_testcraft_yaml_j2, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") project_yaml_path = non_empty_project_dir / project_yaml_filename with check: - assert project_yaml_path.exists(), "Project should be initialized with template" + assert project_yaml_path.exists(), "Project should be initialised with template" project = Project.from_yaml_file(project_yaml_path) with check: assert project.name == fake_project.name -@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_with_different_name_and_directory( init_service: InitService, fake_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, project_yaml_filename: str, emitter, check, ): name = "some-custom-project" - init_service.run(project_dir=fake_empty_project_dir, name=name) + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name=name, + template_dir=template_dir_with_testcraft_yaml_j2, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") project_yaml_path = fake_empty_project_dir / project_yaml_filename with check: - assert project_yaml_path.exists(), "Project should be initialized with template" + assert project_yaml_path.exists(), "Project should be initialised with template" project = Project.from_yaml_file(project_yaml_path) with check: assert project.name == name -@pytest.mark.usefixtures("template_dir_with_testcraft_yaml_j2") def test_init_with_default_arguments_uses_current_directory( init_service: InitService, fake_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, project_yaml_filename: str, emitter, check, ): expected_project_name = fake_empty_project_dir.name - init_service.run(project_dir=fake_empty_project_dir, name="fake-project-dir") + init_service.initialise_project( + project_dir=fake_empty_project_dir, + project_name="fake-project-dir", + template_dir=template_dir_with_testcraft_yaml_j2, + ) with check: - assert emitter.assert_message("Successfully initialised project.") + assert emitter.assert_progress("Rendered project.") project_yaml_path = fake_empty_project_dir / project_yaml_filename with check: - assert project_yaml_path.exists(), "Project should be initialized with template" + assert project_yaml_path.exists(), "Project should be initialised with template" project = Project.from_yaml_file(project_yaml_path) with check: assert project.name == expected_project_name + + +def test_check_for_existing_files( + init_service: InitService, + fake_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, +): + """No-op if there are no overlapping files.""" + init_service.check_for_existing_files( + project_dir=fake_empty_project_dir, + template_dir=template_dir_with_testcraft_yaml_j2, + ) + + +def test_check_for_existing_files_error( + init_service: InitService, + fake_empty_project_dir: pathlib.Path, + template_dir_with_testcraft_yaml_j2: pathlib.Path, +): + """No-op if there are no overlapping files.""" + expected_error = textwrap.dedent( + f"""\ + Cannot initialise project in {str(fake_empty_project_dir)!r} because it would overwrite existing files. + Existing files are: + - nested/testcraft.yaml + - testcraft.yaml""" + ) + (fake_empty_project_dir / "testcraft.yaml").touch() + (fake_empty_project_dir / "nested").mkdir() + (fake_empty_project_dir / "nested" / "testcraft.yaml").touch() + + with pytest.raises(errors.InitError, match=expected_error): + init_service.check_for_existing_files( + project_dir=fake_empty_project_dir, + template_dir=template_dir_with_testcraft_yaml_j2, + ) diff --git a/tests/unit/commands/test_init.py b/tests/unit/commands/test_init.py index 5f0019ee..cbeac193 100644 --- a/tests/unit/commands/test_init.py +++ b/tests/unit/commands/test_init.py @@ -21,38 +21,95 @@ import pytest from craft_application.commands import InitCommand +from craft_application.errors import InitError # init operates in the current working directory pytestmark = pytest.mark.usefixtures("new_dir") @pytest.fixture -def init_command(app_metadata, mock_services): +def init_command(app_metadata, mock_services, mocker, tmp_path): + mocker.patch.object( + InitCommand, + "parent_template_dir", + pathlib.Path(tmp_path) / "templates", + ) return InitCommand({"app": app_metadata, "services": mock_services}) +@pytest.fixture +def fake_template_dirs(tmp_path): + """Set up a fake template directories with two templates. + + These templates are very simple because tests focused on the templates themselves + are in the InitService tests. + """ + parent_template_dir = tmp_path / "templates" + + (parent_template_dir / "simple").mkdir(parents=True) + (parent_template_dir / "other-template").mkdir() + + return parent_template_dir + + @pytest.mark.parametrize("name", [None, "my-project"]) -def test_init_in_cwd(init_command, name, new_dir, mock_services): +def test_init_in_cwd(init_command, name, new_dir, mock_services, emitter): """Test the init command in the current working directory.""" expected_name = name or new_dir.name - parsed_args = argparse.Namespace(project_dir=None, name=name) + parsed_args = argparse.Namespace( + project_dir=None, + name=name, + profile="test-profile", + ) init_command.run(parsed_args) - mock_services.init.run.assert_called_once_with( - project_dir=new_dir, name=expected_name + mock_services.init.initialise_project.assert_called_once_with( + project_dir=new_dir, + project_name=expected_name, + template_dir=init_command.parent_template_dir / "test-profile", ) + emitter.assert_message("Successfully initialised project.") @pytest.mark.parametrize("name", [None, "my-project"]) -def test_init_run_project_dir(init_command, name, mock_services): - """Test the init command in a project directly.""" +def test_init_run_project_dir(init_command, name, mock_services, emitter): + """Test the init command in a project directory.""" expected_name = name or "test-project-dir" project_dir = pathlib.Path("test-project-dir") - parsed_args = argparse.Namespace(project_dir=project_dir, name=name) + parsed_args = argparse.Namespace( + project_dir=project_dir, + name=name, + profile="test-profile", + ) init_command.run(parsed_args) - mock_services.init.run.assert_called_once_with( - project_dir=project_dir, name=expected_name + mock_services.init.initialise_project.assert_called_once_with( + project_dir=project_dir, + project_name=expected_name, + template_dir=init_command.parent_template_dir / "test-profile", + ) + emitter.assert_message("Successfully initialised project.") + + +@pytest.mark.usefixtures("fake_template_dirs") +def test_profiles(init_command): + """Test profile generation.""" + assert init_command.default_profile == "simple" + assert init_command.profiles == ["other-template", "simple"] + + +def test_existing_files(init_command, tmp_path, mock_services): + """Error if the check for existing files fails.""" + mock_services.init.check_for_existing_files.side_effect = InitError("test-error") + parsed_args = argparse.Namespace( + project_dir=tmp_path, + name="test-project-name", + profile="test-profile", ) + + with pytest.raises(InitError, match="test-error"): + init_command.run(parsed_args) + + mock_services.init.initialise_project.assert_not_called() diff --git a/tests/unit/services/test_init.py b/tests/unit/services/test_init.py index 5c640e84..8961dfc9 100644 --- a/tests/unit/services/test_init.py +++ b/tests/unit/services/test_init.py @@ -17,12 +17,12 @@ """Unit tests for the InitService.""" import pathlib -from unittest import mock +import textwrap import jinja2 import pytest import pytest_check -from craft_application import services +from craft_application import errors, services, util @pytest.fixture @@ -33,17 +33,12 @@ def init_service(app_metadata, fake_services): @pytest.fixture -def mock_template(): - _mock_template = mock.Mock(spec=jinja2.Template) - _mock_template.render.return_value = "rendered content" - return _mock_template - - -@pytest.fixture -def mock_environment(mock_template): - _mock_environment = mock.Mock(spec=jinja2.Environment) - _mock_environment.get_template.return_value = mock_template - return _mock_environment +def mock_loader(mocker, tmp_path): + """Mock the loader so it does not try to import `testcraft.templates`.""" + return mocker.patch( + "craft_application.services.init.InitService._get_loader", + return_value=jinja2.FileSystemLoader(tmp_path / "templates"), + ) def test_get_context(init_service): @@ -52,25 +47,13 @@ def test_get_context(init_service): assert context == {"name": "my-project"} -def test_get_template_dir(init_service): - template_dir = init_service._get_template_dir() - - assert template_dir == pathlib.Path("templates") - - -def test_get_executable_files(init_service): - executable_files = init_service._get_executable_files() - - assert executable_files == [] - - def test_create_project_dir(init_service, tmp_path, emitter): project_dir = tmp_path / "my-project" - init_service._create_project_dir(project_dir=project_dir, name="my-project") + init_service._create_project_dir(project_dir=project_dir) assert project_dir.is_dir() - emitter.assert_debug(f"Using project directory {str(project_dir)!r} for my-project") + emitter.assert_debug(f"Creating project directory {str(project_dir)!r}.") def test_create_project_dir_exists(init_service, tmp_path, emitter): @@ -78,10 +61,12 @@ def test_create_project_dir_exists(init_service, tmp_path, emitter): project_dir = tmp_path / "my-project" project_dir.mkdir() - init_service._create_project_dir(project_dir=project_dir, name="my-project") + init_service._create_project_dir(project_dir=project_dir) assert project_dir.is_dir() - emitter.assert_debug(f"Using project directory {str(project_dir)!r} for my-project") + emitter.assert_debug( + f"Not creating project directory {str(project_dir)!r} because it already exists." + ) def test_get_templates_environment(init_service, mocker): @@ -102,6 +87,49 @@ def test_get_templates_environment(init_service, mocker): assert environment == mock_environment.return_value +@pytest.mark.usefixtures("mock_loader") +@pytest.mark.parametrize("project_file", [None, "file.txt"]) +def test_check_for_existing_files(init_service, tmp_path, project_file): + """No-op if there are no overlapping files.""" + # create template + template_dir = tmp_path / "templates" + template_dir.mkdir() + (template_dir / "file.txt").touch() + # create project with a different file + project_dir = tmp_path / "project" + if project_file: + project_dir.mkdir() + (project_dir / "other-file.txt").touch() + + init_service.check_for_existing_files( + project_dir=project_dir, template_dir=template_dir + ) + + +@pytest.mark.usefixtures("mock_loader") +def test_check_for_existing_files_error(init_service, tmp_path): + """Error if there are overlapping files.""" + expected_error = textwrap.dedent( + f"""\ + Cannot initialise project in {str(tmp_path / 'project')!r} because it would overwrite existing files. + Existing files are: + - file.txt""" + ) + # create template + template_dir = tmp_path / "templates" + template_dir.mkdir() + (template_dir / "file.txt").touch() + # create project with a different file + project_dir = tmp_path / "project" + project_dir.mkdir() + (project_dir / "file.txt").touch() + + with pytest.raises(errors.InitError, match=expected_error): + init_service.check_for_existing_files( + project_dir=project_dir, template_dir=template_dir + ) + + @pytest.mark.parametrize("template_filename", ["file1.txt", "nested/file2.txt"]) def test_copy_template_file(init_service, tmp_path, template_filename): # create template @@ -139,67 +167,71 @@ def test_copy_template_file_exists(init_service, tmp_path, template_name, emitte ) -def test_render_project(init_service, tmp_path, mock_environment): - template_filenames = [ - # Jinja2 templates - "jinja-file.txt.j2", - "nested/jinja-file.txt.j2", - # Non-Jinja2 templates (regular files) - "non-jinja-file.txt", - "nested/non-jinja-file.txt", - ] - mock_environment.list_templates.return_value = template_filenames +@pytest.mark.parametrize("filename", ["jinja-file.txt.j2", "nested/jinja-file.txt.j2"]) +@pytest.mark.usefixtures("mock_loader") +def test_render_project_with_templates(filename, init_service, tmp_path): + """Render template files.""" project_dir = tmp_path / "project" project_dir.mkdir() template_dir = tmp_path / "templates" - for filename in template_filenames: - (template_dir / filename).parent.mkdir(parents=True, exist_ok=True) - (template_dir / filename).write_text("template content") + (template_dir / filename).parent.mkdir(parents=True, exist_ok=True) + (template_dir / filename).write_text("{{ name }}") + environment = init_service._get_templates_environment(template_dir) init_service._render_project( - environment=mock_environment, + environment=environment, project_dir=project_dir, template_dir=template_dir, context={"name": "my-project"}, - executable_files=[], ) - pytest_check.equal((project_dir / "jinja-file.txt").read_text(), "rendered content") - pytest_check.equal( - (project_dir / "nested/jinja-file.txt").read_text(), "rendered content" - ) - pytest_check.equal( - (project_dir / "non-jinja-file.txt").read_text(), "template content" - ) - pytest_check.equal( - (project_dir / "nested/non-jinja-file.txt").read_text(), "template content" + assert (project_dir / filename[:-3]).read_text() == "my-project" + + +@pytest.mark.parametrize("filename", ["file.txt", "nested/file.txt"]) +@pytest.mark.usefixtures("mock_loader") +def test_render_project_non_templates(filename, init_service, tmp_path): + """Copy non-template files when rendering a project.""" + project_dir = tmp_path / "project" + project_dir.mkdir() + template_dir = tmp_path / "templates" + (template_dir / filename).parent.mkdir(parents=True, exist_ok=True) + (template_dir / filename).write_text("test content") + + environment = init_service._get_templates_environment(template_dir) + init_service._render_project( + environment=environment, + project_dir=project_dir, + template_dir=template_dir, + context={"name": "my-project"}, ) + assert (project_dir / filename).read_text() == "test content" -def test_render_project_executable(init_service, tmp_path, mock_environment): - template_filenames = [ - "executable-1.sh.j2", - "executable-2.sh.j2", - "executable-3.sh", - ] - mock_environment.list_templates.return_value = template_filenames + +@pytest.mark.usefixtures("mock_loader") +def test_render_project_executable(init_service, tmp_path): + """Test that executable permissions are set on rendered files.""" project_dir = tmp_path / "project" project_dir.mkdir() template_dir = tmp_path / "templates" - for filename in template_filenames: - (template_dir / filename).parent.mkdir(parents=True, exist_ok=True) + template_dir.mkdir() + for filename in ["file-1.sh.j2", "file-2.sh"]: + with (template_dir / filename).open("wt", encoding="utf8") as file: + file.write("#!/bin/bash\necho 'Hello, world!'") + util.make_executable(file) + for filename in ["file-3.txt.j2", "file-4.txt"]: (template_dir / filename).write_text("template content") + environment = init_service._get_templates_environment(template_dir) init_service._render_project( - environment=mock_environment, + environment=environment, project_dir=project_dir, template_dir=template_dir, context={"name": "my-project"}, - executable_files=["executable-1.sh", "executable-3.sh"], ) - import os - - pytest_check.is_true(os.access(project_dir / "executable-1.sh", os.X_OK)) - pytest_check.is_false(os.access(project_dir / "executable-2.sh", os.X_OK)) - pytest_check.is_false(os.access(project_dir / "executable-3.sh", os.X_OK)) + pytest_check.is_true(util.is_executable(project_dir / "file-1.sh")) + pytest_check.is_true(util.is_executable(project_dir / "file-2.sh")) + pytest_check.is_false(util.is_executable(project_dir / "file-3.txt")) + pytest_check.is_false(util.is_executable(project_dir / "file-4.txt")) diff --git a/tests/unit/util/test_file.py b/tests/unit/util/test_file.py index 1c6c860c..c5bc231b 100644 --- a/tests/unit/util/test_file.py +++ b/tests/unit/util/test_file.py @@ -16,7 +16,7 @@ """Unit tests for utility functions and helpers related to path handling.""" -from _stat import S_IRGRP, S_IROTH, S_IRUSR, S_IWUSR, S_IXGRP, S_IXOTH, S_IXUSR +from _stat import S_IEXEC, S_IRGRP, S_IROTH, S_IRUSR, S_IWUSR, S_IXGRP, S_IXOTH, S_IXUSR import pytest from craft_application.util import file @@ -26,6 +26,20 @@ USER_GROUP_OTHER = USER_GROUP | S_IROTH # 0o644 +@pytest.mark.parametrize( + ("mode", "executable"), + [ + (USER, False), + (USER | S_IEXEC, True), + ], +) +def test_is_executable(mode, executable, tmp_path): + test_file = tmp_path / "test-file" + test_file.touch(mode=mode) + + assert file.is_executable(test_file) == executable + + @pytest.mark.parametrize( ("initial_permissions", "expected_permissions"), [ @@ -42,7 +56,7 @@ ), ], ) -def test_make_executable_read_bits(initial_permissions, expected_permissions, tmp_path): +def test_make_executable(initial_permissions, expected_permissions, tmp_path): """make_executable should only operate where the read bits are set.""" test_file = tmp_path / "test" test_file.touch(mode=initial_permissions) From 81ee5d1cda838608fac6c00c90624a6d3ceab08c Mon Sep 17 00:00:00 2001 From: Callahan Kovacs Date: Mon, 21 Oct 2024 08:46:15 -0500 Subject: [PATCH 4/4] chore: feedback from PR Signed-off-by: Callahan Kovacs --- craft_application/commands/init.py | 2 +- craft_application/services/init.py | 18 ++----- craft_application/util/__init__.py | 3 -- craft_application/util/file.py | 51 ------------------- docs/reference/changelog.rst | 2 +- tests/integration/services/test_init.py | 3 +- tests/unit/commands/test_init.py | 2 +- tests/unit/services/test_init.py | 34 +++++-------- tests/unit/util/test_file.py | 68 ------------------------- 9 files changed, 20 insertions(+), 163 deletions(-) delete mode 100644 craft_application/util/file.py delete mode 100644 tests/unit/util/test_file.py diff --git a/craft_application/commands/init.py b/craft_application/commands/init.py index 8395241f..ac954a0d 100644 --- a/craft_application/commands/init.py +++ b/craft_application/commands/init.py @@ -138,7 +138,7 @@ def _get_project_dir(parsed_args: argparse.Namespace) -> pathlib.Path: """ # if set explicitly, just return it if parsed_args.project_dir is not None: - return pathlib.Path(parsed_args.project_dir) + return pathlib.Path(parsed_args.project_dir).expanduser().resolve() # If both args are undefined, default to current dir return pathlib.Path.cwd().resolve() diff --git a/craft_application/services/init.py b/craft_application/services/init.py index 99b55b59..33ee2c88 100644 --- a/craft_application/services/init.py +++ b/craft_application/services/init.py @@ -25,7 +25,6 @@ from craft_cli import emit from craft_application.errors import InitError -from craft_application.util import is_executable, make_executable from . import base @@ -147,11 +146,8 @@ def _render_project( continue path.parent.mkdir(parents=True, exist_ok=True) with path.open("wt", encoding="utf8") as file: - out = template.render(context) - file.write(out) - if is_executable(template_dir / template_name) and os.name == "posix": - make_executable(file) - emit.debug(" made executable") + file.write(template.render(context)) + shutil.copystat((template_dir / template_name), path) emit.progress("Rendered project.") def _get_context(self, name: str) -> dict[str, Any]: @@ -166,14 +162,8 @@ def _get_context(self, name: str) -> dict[str, Any]: @staticmethod def _create_project_dir(project_dir: pathlib.Path) -> None: """Create the project directory if it does not already exist.""" - if not project_dir.exists(): - emit.debug(f"Creating project directory {str(project_dir)!r}.") - project_dir.mkdir(parents=True) - else: - emit.debug( - f"Not creating project directory {str(project_dir)!r} " - "because it already exists." - ) + emit.debug(f"Creating project directory {str(project_dir)!r}.") + project_dir.mkdir(parents=True, exist_ok=True) def _get_loader(self, template_dir: pathlib.Path) -> jinja2.BaseLoader: """Return a Jinja loader for the given template directory. diff --git a/craft_application/util/__init__.py b/craft_application/util/__init__.py index 0498cfb9..6bd33ead 100644 --- a/craft_application/util/__init__.py +++ b/craft_application/util/__init__.py @@ -17,7 +17,6 @@ from craft_application.util.callbacks import get_unique_callbacks from craft_application.util.docs import render_doc_url -from craft_application.util.file import is_executable, make_executable from craft_application.util.logging import setup_loggers from craft_application.util.paths import get_filename_from_url_path, get_managed_logpath from craft_application.util.platforms import ( @@ -39,8 +38,6 @@ __all__ = [ "get_unique_callbacks", "render_doc_url", - "is_executable", - "make_executable", "setup_loggers", "get_filename_from_url_path", "get_managed_logpath", diff --git a/craft_application/util/file.py b/craft_application/util/file.py deleted file mode 100644 index aa1a96dd..00000000 --- a/craft_application/util/file.py +++ /dev/null @@ -1,51 +0,0 @@ -# This file is part of craft_application. -# -# Copyright 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 warranties of MERCHANTABILITY, -# SATISFACTORY QUALITY, 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 . - -"""Helper utilities for files.""" - -import io -import os -import pathlib -from _stat import S_IRGRP, S_IROTH, S_IRUSR - -S_IRALL = S_IRUSR | S_IRGRP | S_IROTH -"""0o444 permission mask for execution permissions for everybody.""" - - -def is_executable(filepath: pathlib.Path) -> bool: - """Check if file is executable. - - :param filepath: The file to check. - - :returns: True if file exists and is executable. - """ - return os.access(filepath, os.X_OK) - - -def make_executable(fh: io.IOBase) -> None: - """Make open file fh executable. - - Only makes the file executable for the user, group, or other if they already had read permissions. - Effectively, this makes the file executable with the same umask as when the file was created. - - :param fh: An open file object. - """ - fileno = fh.fileno() - mode = os.fstat(fileno).st_mode - mode_r = mode & S_IRALL - mode_x = mode_r >> 2 - mode = mode | mode_x - os.fchmod(fileno, mode) diff --git a/docs/reference/changelog.rst b/docs/reference/changelog.rst index 0e5ad8ac..4fa598c9 100644 --- a/docs/reference/changelog.rst +++ b/docs/reference/changelog.rst @@ -4,7 +4,7 @@ Changelog ********* -X.Y.Z (2024-MMM-DD) +X.Y.0 (2024-MMM-DD) ------------------- Commands diff --git a/tests/integration/services/test_init.py b/tests/integration/services/test_init.py index a75b5a7b..c911cd44 100644 --- a/tests/integration/services/test_init.py +++ b/tests/integration/services/test_init.py @@ -25,7 +25,6 @@ from craft_application import errors from craft_application.models.project import Project from craft_application.services import InitService -from craft_application.util import make_executable # init operates in the current working directory pytestmark = pytest.mark.usefixtures("new_dir") @@ -145,7 +144,7 @@ def template_dir_with_executables( filepath.parent.mkdir(exist_ok=True) with filepath.open("wt", encoding="utf8") as file: file.write("#!/bin/bash\necho 'Hello, world!'") - make_executable(file) + filepath.chmod(0o755) return fake_empty_template_dir diff --git a/tests/unit/commands/test_init.py b/tests/unit/commands/test_init.py index cbeac193..3f92f596 100644 --- a/tests/unit/commands/test_init.py +++ b/tests/unit/commands/test_init.py @@ -86,7 +86,7 @@ def test_init_run_project_dir(init_command, name, mock_services, emitter): init_command.run(parsed_args) mock_services.init.initialise_project.assert_called_once_with( - project_dir=project_dir, + project_dir=project_dir.expanduser().resolve(), project_name=expected_name, template_dir=init_command.parent_template_dir / "test-profile", ) diff --git a/tests/unit/services/test_init.py b/tests/unit/services/test_init.py index 8961dfc9..bd9cdca1 100644 --- a/tests/unit/services/test_init.py +++ b/tests/unit/services/test_init.py @@ -16,13 +16,14 @@ """Unit tests for the InitService.""" +import os import pathlib import textwrap import jinja2 import pytest import pytest_check -from craft_application import errors, services, util +from craft_application import errors, services @pytest.fixture @@ -47,8 +48,11 @@ def test_get_context(init_service): assert context == {"name": "my-project"} -def test_create_project_dir(init_service, tmp_path, emitter): +@pytest.mark.parametrize("create_dir", [True, False]) +def test_create_project_dir(init_service, tmp_path, emitter, create_dir): project_dir = tmp_path / "my-project" + if create_dir: + project_dir.mkdir() init_service._create_project_dir(project_dir=project_dir) @@ -56,19 +60,6 @@ def test_create_project_dir(init_service, tmp_path, emitter): emitter.assert_debug(f"Creating project directory {str(project_dir)!r}.") -def test_create_project_dir_exists(init_service, tmp_path, emitter): - """Do not error if the project directory already exists.""" - project_dir = tmp_path / "my-project" - project_dir.mkdir() - - init_service._create_project_dir(project_dir=project_dir) - - assert project_dir.is_dir() - emitter.assert_debug( - f"Not creating project directory {str(project_dir)!r} because it already exists." - ) - - def test_get_templates_environment(init_service, mocker): """Test that _get_templates_environment returns a Jinja2 environment.""" mock_package_loader = mocker.patch("jinja2.PackageLoader") @@ -217,9 +208,8 @@ def test_render_project_executable(init_service, tmp_path): template_dir = tmp_path / "templates" template_dir.mkdir() for filename in ["file-1.sh.j2", "file-2.sh"]: - with (template_dir / filename).open("wt", encoding="utf8") as file: - file.write("#!/bin/bash\necho 'Hello, world!'") - util.make_executable(file) + (template_dir / filename).write_text("#!/bin/bash\necho 'Hello, world!'") + (template_dir / filename).chmod(0o755) for filename in ["file-3.txt.j2", "file-4.txt"]: (template_dir / filename).write_text("template content") @@ -231,7 +221,7 @@ def test_render_project_executable(init_service, tmp_path): context={"name": "my-project"}, ) - pytest_check.is_true(util.is_executable(project_dir / "file-1.sh")) - pytest_check.is_true(util.is_executable(project_dir / "file-2.sh")) - pytest_check.is_false(util.is_executable(project_dir / "file-3.txt")) - pytest_check.is_false(util.is_executable(project_dir / "file-4.txt")) + pytest_check.is_true(os.access(project_dir / "file-1.sh", os.X_OK)) + pytest_check.is_true(os.access(project_dir / "file-2.sh", os.X_OK)) + pytest_check.is_false(os.access(project_dir / "file-3.txt", os.X_OK)) + pytest_check.is_false(os.access(project_dir / "file-4.txt", os.X_OK)) diff --git a/tests/unit/util/test_file.py b/tests/unit/util/test_file.py deleted file mode 100644 index c5bc231b..00000000 --- a/tests/unit/util/test_file.py +++ /dev/null @@ -1,68 +0,0 @@ -# This file is part of craft_application. -# -# Copyright 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 warranties of MERCHANTABILITY, -# SATISFACTORY QUALITY, 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 . - -"""Unit tests for utility functions and helpers related to path handling.""" - -from _stat import S_IEXEC, S_IRGRP, S_IROTH, S_IRUSR, S_IWUSR, S_IXGRP, S_IXOTH, S_IXUSR - -import pytest -from craft_application.util import file - -USER = S_IRUSR | S_IWUSR # 0o600 -USER_GROUP = USER | S_IRGRP # 0o640 -USER_GROUP_OTHER = USER_GROUP | S_IROTH # 0o644 - - -@pytest.mark.parametrize( - ("mode", "executable"), - [ - (USER, False), - (USER | S_IEXEC, True), - ], -) -def test_is_executable(mode, executable, tmp_path): - test_file = tmp_path / "test-file" - test_file.touch(mode=mode) - - assert file.is_executable(test_file) == executable - - -@pytest.mark.parametrize( - ("initial_permissions", "expected_permissions"), - [ - pytest.param(USER, USER | S_IXUSR, id="user"), - pytest.param( - USER_GROUP, - USER_GROUP | S_IXUSR | S_IXGRP, - id="user-group", - ), - pytest.param( - USER_GROUP_OTHER, - USER_GROUP_OTHER | S_IXUSR | S_IXGRP | S_IXOTH, - id="user-group-other", - ), - ], -) -def test_make_executable(initial_permissions, expected_permissions, tmp_path): - """make_executable should only operate where the read bits are set.""" - test_file = tmp_path / "test" - test_file.touch(mode=initial_permissions) - - with test_file.open() as fd: - file.make_executable(fd) - - # only read bits got made executable - assert test_file.stat().st_mode & 0o777 == expected_permissions