From 283250900e7d8dee4271d7891676ecb4c72bd122 Mon Sep 17 00:00:00 2001 From: Alex Lowe Date: Mon, 9 Sep 2024 16:20:41 -0400 Subject: [PATCH] fix: lock shared cache directory Locks the shared cache directory to prevent concurrency issues. Fixes #1845 CRAFT-3313 --- charmcraft/services/provider.py | 36 +++++++++- tests/conftest.py | 21 +++--- tests/integration/services/conftest.py | 17 +---- tests/integration/services/test_package.py | 36 ++++------ tests/integration/services/test_provider.py | 49 +++++++++++++ tests/unit/services/test_provider.py | 80 +++++++++++++++++++++ 6 files changed, 192 insertions(+), 47 deletions(-) create mode 100644 tests/integration/services/test_provider.py diff --git a/charmcraft/services/provider.py b/charmcraft/services/provider.py index cefcb9ad9..958a2b9ee 100644 --- a/charmcraft/services/provider.py +++ b/charmcraft/services/provider.py @@ -17,11 +17,15 @@ """Service class for creating providers.""" from __future__ import annotations +import atexit +import fcntl import os import pathlib +from typing import cast import craft_providers from craft_application import services +from craft_cli import emit from craft_providers import bases from charmcraft import env @@ -56,12 +60,40 @@ def get_base( If no cache_path is included, adds one. """ + cache_path = cast(pathlib.Path, kwargs.get("cache_path", env.get_host_shared_cache_path())) + our_lock = _maybe_lock_cache(cache_path) + # Forward the shared cache path. - if "cache_path" not in kwargs: - kwargs["cache_path"] = env.get_host_shared_cache_path() + kwargs["cache_path"] = cache_path if our_lock else None return super().get_base( base_name, instance_name=instance_name, # craft-application annotation is incorrect **kwargs, # type: ignore[arg-type] ) + + +def _maybe_lock_cache(path: pathlib.Path) -> bool: + """Lock the cache so we only have one copy of Charmcraft using it at a time.""" + cache_lock_path = path / "charmcraft.lock" + + emit.trace("Attempting to lock the cache path") + lock_file = cache_lock_path.open("w+") + try: + # Exclusive lock, but non-blocking. + fcntl.flock(lock_file, fcntl.LOCK_EX | fcntl.LOCK_NB) + except OSError: + emit.progress( + "Shared cache locked by another process; running without cache.", permanent=True + ) + return False + else: + pid = str(os.getpid()) + lock_file.write(pid) + lock_file.flush() + os.fsync(lock_file.fileno()) + emit.trace(f"Cache path locked by this process ({pid})") + atexit.register(fcntl.flock, lock_file, fcntl.LOCK_UN) + atexit.register(cache_lock_path.unlink, missing_ok=True) + + return True diff --git a/tests/conftest.py b/tests/conftest.py index 1eeecf33d..8b66dd28f 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -109,16 +109,19 @@ def service_factory( @pytest.fixture -def default_build_plan(): +def default_build_info() -> models.BuildInfo: arch = util.get_host_architecture() - return [ - models.BuildInfo( - base=bases.BaseName("ubuntu", "22.04"), - build_on=arch, - build_for="arm64", - platform="distro-1-test64", - ) - ] + return models.BuildInfo( + base=bases.BaseName("ubuntu", "22.04"), + build_on=arch, + build_for="arm64", + platform="distro-1-test64", + ) + + +@pytest.fixture +def default_build_plan(default_build_info: models.BuildInfo): + return [default_build_info] @pytest.fixture diff --git a/tests/integration/services/conftest.py b/tests/integration/services/conftest.py index 04704da92..96f232f06 100644 --- a/tests/integration/services/conftest.py +++ b/tests/integration/services/conftest.py @@ -14,10 +14,7 @@ # # For further info, check https://github.com/canonical/charmcraft """Configuration for services integration tests.""" -import contextlib -import sys -import pyfakefs.fake_filesystem import pytest from charmcraft import services @@ -25,20 +22,10 @@ @pytest.fixture -def service_factory( - fs: pyfakefs.fake_filesystem.FakeFilesystem, fake_path, simple_charm -) -> services.CharmcraftServiceFactory: - fake_project_dir = fake_path / "project" +def service_factory(simple_charm, new_path) -> services.CharmcraftServiceFactory: + fake_project_dir = new_path / "project" fake_project_dir.mkdir() - # Allow access to the real venv library path. - # This is necessary because certifi lazy-loads the certificate file. - for python_path in sys.path: - if not python_path: - continue - with contextlib.suppress(OSError): - fs.add_real_directory(python_path) - factory = services.CharmcraftServiceFactory(app=APP_METADATA) app = Charmcraft(app=APP_METADATA, services=factory) diff --git a/tests/integration/services/test_package.py b/tests/integration/services/test_package.py index cfe9fc0b8..a2bccce96 100644 --- a/tests/integration/services/test_package.py +++ b/tests/integration/services/test_package.py @@ -28,8 +28,8 @@ @pytest.fixture -def package_service(fake_path, service_factory, default_build_plan): - fake_project_dir = fake_path +def package_service(new_path: pathlib.Path, service_factory, default_build_plan): + fake_project_dir = new_path svc = services.PackageService( app=APP_METADATA, project=service_factory.project, @@ -49,12 +49,10 @@ def package_service(fake_path, service_factory, default_build_plan): ], ) @freezegun.freeze_time(datetime.datetime(2020, 3, 14, 0, 0, 0, tzinfo=datetime.timezone.utc)) -def test_write_metadata(monkeypatch, fs, package_service, project_path): +def test_write_metadata(monkeypatch, new_path, package_service, project_path): monkeypatch.setattr(charmcraft, "__version__", "3.0-test-version") - with contextlib.suppress(FileExistsError): - fs.add_real_directory(project_path) - test_prime_dir = pathlib.Path("/prime") - fs.create_dir(test_prime_dir) + test_prime_dir = new_path / "prime" + test_prime_dir.mkdir() expected_prime_dir = project_path / "prime" project = models.CharmcraftProject.from_yaml_file(project_path / "project" / "charmcraft.yaml") @@ -75,23 +73,21 @@ def test_write_metadata(monkeypatch, fs, package_service, project_path): ], ) @freezegun.freeze_time(datetime.datetime(2020, 3, 14, 0, 0, 0, tzinfo=datetime.timezone.utc)) -def test_overwrite_metadata(monkeypatch, fs, package_service, project_path): +def test_overwrite_metadata(monkeypatch, new_path, package_service, project_path): """Test that the metadata file gets rewritten for a charm. Regression test for https://github.com/canonical/charmcraft/issues/1654 """ monkeypatch.setattr(charmcraft, "__version__", "3.0-test-version") - with contextlib.suppress(FileExistsError): - fs.add_real_directory(project_path) - test_prime_dir = pathlib.Path("/prime") - fs.create_dir(test_prime_dir) + test_prime_dir = new_path / "prime" + test_prime_dir.mkdir() expected_prime_dir = project_path / "prime" project = models.CharmcraftProject.from_yaml_file(project_path / "project" / "charmcraft.yaml") project._started_at = datetime.datetime.now(tz=datetime.timezone.utc) package_service._project = project - fs.create_file(test_prime_dir / const.METADATA_FILENAME, contents="INVALID!!") + (test_prime_dir / const.METADATA_FILENAME).write_text("INVALID!!") package_service.write_metadata(test_prime_dir) @@ -100,20 +96,18 @@ def test_overwrite_metadata(monkeypatch, fs, package_service, project_path): @freezegun.freeze_time(datetime.datetime(2020, 3, 14, 0, 0, 0, tzinfo=datetime.timezone.utc)) -def test_no_overwrite_reactive_metadata(monkeypatch, fs, package_service): +def test_no_overwrite_reactive_metadata(monkeypatch, new_path, package_service): """Test that the metadata file doesn't get overwritten for a reactive charm.. Regression test for https://github.com/canonical/charmcraft/issues/1654 """ monkeypatch.setattr(charmcraft, "__version__", "3.0-test-version") project_path = pathlib.Path(__file__).parent / "sample_projects" / "basic-reactive" - with contextlib.suppress(FileExistsError): - fs.add_real_directory(project_path) - test_prime_dir = pathlib.Path("/prime") - fs.create_dir(test_prime_dir) - test_stage_dir = pathlib.Path("/stage") - fs.create_dir(test_stage_dir) - fs.create_file(test_stage_dir / const.METADATA_FILENAME, contents="INVALID!!") + test_prime_dir = new_path / "prime" + test_prime_dir.mkdir() + test_stage_dir = new_path / "stage" + test_stage_dir.mkdir() + (test_stage_dir / const.METADATA_FILENAME).write_text("INVALID!!") project = models.CharmcraftProject.from_yaml_file(project_path / "project" / "charmcraft.yaml") project._started_at = datetime.datetime.now(tz=datetime.timezone.utc) diff --git a/tests/integration/services/test_provider.py b/tests/integration/services/test_provider.py new file mode 100644 index 000000000..2c05c6416 --- /dev/null +++ b/tests/integration/services/test_provider.py @@ -0,0 +1,49 @@ +# Copyright 2024 Canonical Ltd. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# For further info, check https://github.com/canonical/charmcraft +"""Integration tests for the provider service.""" + +import pathlib + +from craft_application.models import BuildInfo +from craft_cli.pytest_plugin import RecordingEmitter + +from charmcraft import services +from charmcraft.services.provider import _maybe_lock_cache + + +def test_locks_cache( + service_factory: services.CharmcraftServiceFactory, + tmp_path: pathlib.Path, + default_build_info: BuildInfo, + emitter: RecordingEmitter, +): + _maybe_lock_cache(tmp_path) + assert (tmp_path / "charmcraft.lock").exists() + provider = service_factory.provider + provider_kwargs = { + "build_info": default_build_info, + "work_dir": pathlib.Path(__file__), + "cache_path": tmp_path, + } + + with provider.instance(**provider_kwargs) as instance: + # Because we've already locked the cache, we shouldn't see the lockfile. + lock_test = instance.execute_run(["test", "-f", "/root/.cache/charmcraft.lock"]) + assert lock_test.returncode == 1 + + # Create a file in the cache and ensure it's not visible in the outer fs + instance.execute_run(["touch", "/root/.cache/cache_cached"]) + assert not (tmp_path / "cache_cached").exists() diff --git a/tests/unit/services/test_provider.py b/tests/unit/services/test_provider.py index b55f6c48d..02540e8b5 100644 --- a/tests/unit/services/test_provider.py +++ b/tests/unit/services/test_provider.py @@ -15,12 +15,19 @@ # For further info, check https://github.com/canonical/charmcraft """Unit tests for the provider service.""" +import fcntl +import functools import pathlib +from collections.abc import Iterator +from unittest import mock import pytest +from craft_cli.pytest_plugin import RecordingEmitter from craft_providers import bases from charmcraft import models, services +from charmcraft.application.main import APP_METADATA +from charmcraft.services.provider import _maybe_lock_cache @pytest.fixture @@ -42,6 +49,17 @@ def provider_service( return service_factory.provider +@pytest.fixture +def mock_register(monkeypatch) -> Iterator[mock.Mock]: + register = mock.Mock() + monkeypatch.setattr("atexit.register", register) + yield register + + # Call the exit hooks as if exiting the application. + for hook in register.mock_calls: + functools.partial(*hook.args)() + + @pytest.mark.parametrize( "base_name", [ @@ -76,3 +94,65 @@ def test_get_base_forwards_cache( ) assert base._cache_path == fake_path / "cache" + + +@pytest.mark.parametrize( + "base_name", + [ + bases.BaseName("ubuntu", "20.04"), + bases.BaseName("ubuntu", "22.04"), + bases.BaseName("ubuntu", "24.04"), + bases.BaseName("ubuntu", "devel"), + bases.BaseName("almalinux", "9"), + ], +) +def test_get_base_no_cache_if_locked( + monkeypatch, + mock_register, + tmp_path: pathlib.Path, + base_name: bases.BaseName, + emitter: RecordingEmitter, +): + cache_path = tmp_path / "cache" + cache_path.mkdir(exist_ok=True, parents=True) + + # Make a new path object to work around caching the paths and thus getting the + # same file descriptor. + locked = _maybe_lock_cache(cache_path) + assert locked + new_cache_path = pathlib.Path(str(cache_path)) + monkeypatch.setattr("charmcraft.env.get_host_shared_cache_path", lambda: new_cache_path) + + # Can't use the fixture as pyfakefs doesn't handle locks. + provider_service = services.ProviderService( + app=APP_METADATA, + services=None, # pyright: ignore[reportArgumentType] + project=None, # pyright: ignore[reportArgumentType] + work_dir=tmp_path, + build_plan=[], + ) + + base = provider_service.get_base( + base_name=base_name, + instance_name="charmcraft-test-instance", + ) + + assert base._cache_path is None + emitter.assert_progress( + "Shared cache locked by another process; running without cache.", + permanent=True, + ) + + +def test_maybe_lock_cache_locks_single_lock(tmp_path: pathlib.Path, mock_register) -> None: + assert _maybe_lock_cache(tmp_path) is True + mock_register.assert_has_calls([mock.call(fcntl.flock, mock.ANY, fcntl.LOCK_UN)]) + + +def test_maybe_lock_cache_with_another_lock(tmp_path: pathlib.Path, mock_register) -> None: + assert _maybe_lock_cache(tmp_path) is True + assert _maybe_lock_cache(tmp_path) is False + assert mock_register.mock_calls == [ + mock.call(fcntl.flock, mock.ANY, fcntl.LOCK_UN), + mock.call(mock.ANY, missing_ok=True), + ]