From 49a7c05d61a27c11c10de3479214a1d8abceaea0 Mon Sep 17 00:00:00 2001 From: Adam Simpkins Date: Wed, 31 Oct 2018 12:08:27 -0700 Subject: [PATCH] add code to test saved snapshots Summary: Update the snapshot framework code to add tests that mount saved snapshots with the current edenfs build and verifies that they behave as expected. This also changes the snapshot generation code a fair amount as well, in order to be able to verify the snapshot data: - We now more cleanly split out data that should be saved when the snapshot is first generated (such as the `.eden` state directory) and data that should be re-generated each time the snapshot is unpacked (such as the `/etc/eden` config directory and the system configuration for `hg`). - The code is now capable of rewriting absolute paths in the Eden state files so it is usable in a new location after it has been unpacked. This also updates the "basic" snapshot type to create a wider variety of types of files and file changes, and to be able to verify the contents of a resumed snapshot. Reviewed By: strager Differential Revision: D9955411 fbshipit-source-id: 76012c01016cf4d89dc611c5596a05f5e7f013ed --- eden/integration/snapshot/snapshot.py | 372 +++++++++++++++--- eden/integration/snapshot/test_snapshots.py | 131 ++++++ eden/integration/snapshot/types/basic.py | 135 ++++++- eden/integration/snapshot/verify.py | 174 ++++++++ .../test-data/snapshots/basic-20181030.tar.xz | Bin 0 -> 16508 bytes 5 files changed, 746 insertions(+), 66 deletions(-) create mode 100644 eden/integration/snapshot/test_snapshots.py create mode 100644 eden/integration/snapshot/verify.py create mode 100644 eden/test-data/snapshots/basic-20181030.tar.xz diff --git a/eden/integration/snapshot/snapshot.py b/eden/integration/snapshot/snapshot.py index d5104ccd30..df03de482b 100644 --- a/eden/integration/snapshot/snapshot.py +++ b/eden/integration/snapshot/snapshot.py @@ -9,23 +9,32 @@ import abc import contextlib +import datetime import json import logging +import os +import socket +import stat import subprocess import tempfile import time import types +import typing from pathlib import Path -from typing import Callable, Dict, Iterator, List, Optional, Type, TypeVar, Union +from typing import Any, Callable, Dict, Iterator, List, Optional, Type, TypeVar, Union +import toml from eden.integration.lib import edenclient, hgrepo, util +from eden.integration.lib.find_executables import FindExe from eden.integration.lib.temporary_directory import create_tmp_dir +from . import verify as verify_mod + T = TypeVar("T", bound="BaseSnapshot") -class BaseSnapshot: +class BaseSnapshot(metaclass=abc.ABCMeta): # The NAME and DESCRIPTION class fields are intended to be overridden on subclasses # by the @snapshot_class decorator. NAME = "Base Snapshot Class" @@ -33,7 +42,22 @@ class BaseSnapshot: def __init__(self, base_dir: Path) -> None: self.base_dir = base_dir - self.eden: Optional[edenclient.EdenFS] = None + # All data inside self.data_dir will be saved as part of the snapshot + self.data_dir = self.base_dir / "data" + # Anything inside self.transient_dir will not be saved with the snapshot, + # and will always be regenerated from scratch when resuming a snapshot. + self.transient_dir = self.base_dir / "transient" + + self.eden_state_dir = self.data_dir / "eden" + + # We put the etc eden directory inside the transient directory. + # Whenever we resume a snapshot we want to use a current version of the edenfs + # daemon and its configuration, rather than an old copy of the edenfs + # configuration. + self.etc_eden_dir = self.transient_dir / "etc_eden" + + # We put the home directory inside the transient directory as well. + self.home_dir = self.transient_dir / "home" def __enter__(self: T) -> T: return self @@ -44,15 +68,7 @@ class BaseSnapshot: exc_value: Optional[BaseException], tb: Optional[types.TracebackType], ) -> None: - self.cleanup() - - def cleanup(self) -> None: - if self.eden is not None: - try: - self.eden.kill() - except Exception as ex: - logging.exception("error stopping edenfs") - self.eden = None + pass def create_tarball(self, output_path: Path) -> None: """Create a tarball from the snapshot contents. @@ -85,46 +101,209 @@ class BaseSnapshot: subprocess.check_call(cmd, cwd=self.base_dir) def generate(self) -> None: - self._setup_directories() + """Generate the snapshot data. + + This method should normally be called after constructing the snapshot object + pointing to an empty directory. + """ + self._create_directories() self._emit_metadata() self.gen_before_eden_running() - self.eden = edenclient.EdenFS( + with self.edenfs() as eden: + eden.start() + self.gen_eden_running(eden) + + self.gen_after_eden_stopped() + + # Rewrite the config state to point to "/tmp/dummy_snapshot_path" + # This isn't really strictly necessary, but just makes the state that + # gets saved slightly more deterministic. + self._relocate_to(Path("/tmp/dummy_snapshot_path")) + + def verify(self, verifier: verify_mod.SnapshotVerifier) -> None: + """Verify that the snapshot data looks correct. + + This is generally invoked by tests to confirm that an unpacked snapshot still + works properly with the current version of EdenFS. + """ + with self.edenfs() as eden: + eden.start() + print("Verifing snapshot data:") + print("=" * 60) + self.verify_snapshot_data(verifier, eden) + print("=" * 60) + + def edenfs(self) -> edenclient.EdenFS: + """Return an EdenFS object that can be used to run an edenfs daemon for this + snapshot. + + The returned EdenFS object will not be started yet; the caller must explicitly + call start() on it. + """ + return edenclient.EdenFS( eden_dir=str(self.eden_state_dir), etc_eden_dir=str(self.etc_eden_dir), home_dir=str(self.home_dir), storage_engine="rocksdb", ) - try: - self.eden.start() - self.gen_eden_running() - finally: - self.eden.kill() - self.eden = None - self.gen_after_eden_stopped() + def resume(self) -> None: + """Prepare a snapshot to be resumed after unpacking it. - def _setup_directories(self) -> None: - self.data_dir = self.base_dir / "data" + This updates the snapshot data so it can be run from its new location, + and recreates any transient state needed for the snapshot. + """ + self.create_transient_dir() + self._relocate_to(self.base_dir) + self.prep_resume() + + def _create_directories(self) -> None: self.data_dir.mkdir() + self.create_transient_dir() - self.eden_state_dir = self.data_dir / "eden" - self.etc_eden_dir = self.data_dir / "etc_eden" + def create_transient_dir(self) -> None: + self.transient_dir.mkdir() self.etc_eden_dir.mkdir() - self.home_dir = self.data_dir / "home" self.home_dir.mkdir() + # Set up configuration and hooks inside the etc eden directory. + hooks_dir = self.etc_eden_dir / "hooks" + hooks_dir.mkdir() + os.symlink(FindExe.EDEN_POST_CLONE_HOOK, hooks_dir / "post-clone") + config_dir = self.etc_eden_dir / "config.d" + config_dir.mkdir() + + # Set the hg.edenextension path to the empty string, so that + # we use the version of the eden extension built into hg.par + toml_config = {"hooks": {"hg.edenextension": ""}} + with (config_dir / "hooks").open("w") as f: + toml.dump(toml_config, f) + def _emit_metadata(self) -> None: + now = time.time() + + # In addition to recording the current time as a unix timestamp, + # we also store a tuple of (year, month, day). This is primarily to help make + # it easier for future verification code if we ever need to alter the + # verification logic for older versions of the same snapshot type. + # This will allow more human-readable time comparisons in the code, and makes it + # easier to compare just based on a prefix of this tuple. + now_date = datetime.datetime.fromtimestamp(now) + date_tuple = ( + now_date.year, + now_date.month, + now_date.day, + now_date.hour, + now_date.minute, + now_date.second, + ) + data = { "type": self.NAME, "description": self.DESCRIPTION, - "time_created": time.time(), + "time_created": int(now), + "date_created": date_tuple, + "base_dir": str(self.base_dir), } + self._write_metadata(data) - metadata_path = self.data_dir / "info.json" - with metadata_path.open("w") as f: + @property + def _metadata_path(self) -> Path: + return self.data_dir / "info.json" + + def _write_metadata(self, data: Dict[str, Any]) -> None: + with self._metadata_path.open("w") as f: json.dump(data, f, indent=2, sort_keys=True) + def _read_metadata(self) -> Dict[str, Any]: + with self._metadata_path.open("r") as f: + return typing.cast(Dict[str, Any], json.load(f)) + + def _relocate_to(self, base_dir: Path) -> None: + """Rewrite data inside an unpacked snapshot directory to refer to the base + directory using the specified path. + + This replaces absolute path names in various data files to refer to the new + location. This is needed so that a snapshot originally created in one location + can be unpacked and used in another location. + """ + info = self._read_metadata() + old_base_dir = Path(info["base_dir"]) + + # A few files in the RocksDB directory end up with the absolute path + # embedded in them. + rocks_db_path = self.eden_state_dir / "storage" / "rocks-db" + for entry in rocks_db_path.iterdir(): + if entry.name.startswith("LOG") or entry.name.startswith("OPTIONS"): + self._replace_file_contents(entry, bytes(old_base_dir), bytes(base_dir)) + + # Parse eden's config.json to get the list of checkouts, and update each one. + eden_config_path = self.eden_state_dir / "config.json" + with eden_config_path.open("r+") as config_file: + eden_data = json.load(config_file) + new_config_data = {} + for _old_checkout_path, checkout_name in eden_data.items(): + new_checkout_path = self.data_dir / checkout_name + new_config_data[str(new_checkout_path)] = checkout_name + checkout_state_dir = self.eden_state_dir / "clients" / checkout_name + self._relocate_checkout(checkout_state_dir, old_base_dir, base_dir) + + config_file.seek(0) + config_file.truncate() + json.dump(new_config_data, config_file, indent=2, sort_keys=True) + + # Update the info file with the new base path + info["base_dir"] = str(base_dir) + self._write_metadata(info) + + def _relocate_checkout( + self, checkout_state_dir: Path, old_base_dir: Path, new_base_dir: Path + ) -> None: + self._replace_file_contents( + checkout_state_dir / "config.toml", bytes(old_base_dir), bytes(new_base_dir) + ) + overlay_dir = checkout_state_dir / "local" + self._relocate_overlay_dir( + overlay_dir, bytes(old_base_dir), bytes(new_base_dir) + ) + + def _relocate_overlay_dir( + self, dir_path: Path, old_data: bytes, new_data: bytes + ) -> None: + # Recursively update the contents for every file in the overlay + # if it contains the old path. + # + # This approach is pretty dumb: we aren't processing the overlay file formats at + # all, just blindly replacing the contents if we happen to see something that + # looks like the old path. For now this is the easiest thing to do, and the + # chance of other data looking like the source path should be very unlikely. + # + # In practice we normally need to update the overlay files for at least the + # following inodes: + # .eden/root + # .eden/client + # .eden/socket + # .hg/sharedpath + # + for path in dir_path.iterdir(): + stat_info = path.lstat() + if stat.S_ISDIR(stat_info.st_mode): + self._relocate_overlay_dir(path, old_data, new_data) + else: + self._replace_file_contents(path, old_data, new_data) + + def _replace_file_contents( + self, path: Path, old_data: bytes, new_data: bytes + ) -> None: + with path.open("rb+") as f: + file_contents = f.read() + new_contents = file_contents.replace(old_data, new_data) + if new_contents != file_contents: + f.seek(0) + f.truncate() + f.write(new_contents) + def gen_before_eden_running(self) -> None: """gen_before_eden_running() will be called when generating a new snapshot after the directory structure has been set up but before edenfs is started. @@ -133,7 +312,7 @@ class BaseSnapshot: """ pass - def gen_eden_running(self) -> None: + def gen_eden_running(self, eden: edenclient.EdenFS) -> None: """gen_eden_running() will be called when generating a new snapshot once edenfs has been started. @@ -149,37 +328,60 @@ class BaseSnapshot: """ pass + def prep_resume(self) -> None: + """prep_resume() will be when preparing to resume a snapshot, before edenfs has + been started. + + Subclasses of BaseSnapshot can perform any work they want here. + here. + """ + pass + + @abc.abstractmethod + def verify_snapshot_data( + self, verifier: verify_mod.SnapshotVerifier, eden: edenclient.EdenFS + ) -> None: + """Verify that the snapshot data looks correct. + + This method should be overridden by subclasses. + """ + pass + class HgSnapshot(BaseSnapshot, metaclass=abc.ABCMeta): """A helper parent class for BaseSnapshot implementations that creates a single checkout of a mercurial repository.""" - def gen_before_eden_running(self) -> None: - # Prepare the system hgrc file - self.system_hgrc_path = self.data_dir / "system_hgrc" + def create_transient_dir(self) -> None: + super().create_transient_dir() + + # Note that we put the system hgrc file in self.transient_dir rather than + # self.data_dir: + # This file is not saved with the snapshot, and is instead regenerated each time + # we unpack the snapshot. This reflects the fact that we always run with the + # current system hgrc rather than an old snapshot of the system configs. + self.system_hgrc_path = self.transient_dir / "system_hgrc" self.system_hgrc_path.write_text(hgrepo.HgRepository.get_system_hgrc_contents()) + def hg_repo(self, path: Path) -> hgrepo.HgRepository: + return hgrepo.HgRepository(str(path), system_hgrc=str(self.system_hgrc_path)) + + def gen_before_eden_running(self) -> None: logging.info("Creating backing repository...") # Create the repository backing_repo_path = self.data_dir / "repo" backing_repo_path.mkdir() - self.backing_repo = hgrepo.HgRepository( - str(backing_repo_path), system_hgrc=str(self.system_hgrc_path) - ) + self.backing_repo = self.hg_repo(backing_repo_path) self.backing_repo.init() self.populate_backing_repo() - def gen_eden_running(self) -> None: - assert self.eden is not None + def gen_eden_running(self, eden: edenclient.EdenFS) -> None: logging.info("Preparing checkout...") - checkout_path = self.data_dir / "checkout" - self.eden.clone(self.backing_repo.path, str(checkout_path)) + eden.clone(self.backing_repo.path, str(self.checkout_path)) - self.checkout_repo = hgrepo.HgRepository( - str(checkout_path), system_hgrc=str(self.system_hgrc_path) - ) + self.checkout_repo = self.hg_repo(self.checkout_path) self.populate_checkout() @abc.abstractmethod @@ -190,32 +392,62 @@ class HgSnapshot(BaseSnapshot, metaclass=abc.ABCMeta): def populate_checkout(self) -> None: pass - def checkout_path(self, *args: Union[Path, str]) -> Path: - """Compute a path inside the checkout.""" - return Path(self.checkout_repo.path, *args) + @property + def checkout_path(self) -> Path: + """Return the path to the checkout root.""" + return self.data_dir / "checkout" def read_file(self, path: Union[Path, str]) -> bytes: """Helper function to read a file in the checkout. This is primarily used to ensure that the file is loaded. """ - file_path = self.checkout_path(path) + file_path = self.checkout_path / path with file_path.open("rb") as f: data: bytes = f.read() return data - def write_file(self, path: Union[Path, str], contents: bytes) -> None: + def write_file( + self, path: Union[Path, str], contents: bytes, mode: int = 0o644 + ) -> None: """Helper function to write a file in the checkout.""" - file_path = self.checkout_path(path) + file_path = self.checkout_path / path + file_path.parent.mkdir(parents=True, exist_ok=True) with file_path.open("wb") as f: + os.fchmod(f.fileno(), mode) f.write(contents) + def chmod(self, path: Union[Path, str], mode: int) -> None: + file_path = self.checkout_path / path + os.chmod(file_path, mode) + + def mkdir(self, path: Union[Path, str], mode: int = 0o755) -> None: + dir_path = self.checkout_path / path + dir_path.mkdir(mode=mode, parents=True, exist_ok=False) + # Explicitly call chmod() to ignore any umask settings + dir_path.chmod(mode) + def list_dir(self, path: Union[Path, str]) -> List[Path]: """List the contents of a directory in the checkout. This can be used to ensure the directory has been loaded by Eden. """ - dir_path = self.checkout_path(path) + dir_path = self.checkout_path / path return list(dir_path.iterdir()) + def make_socket(self, path: Union[Path, str], mode: int = 0o755) -> None: + socket_path = self.checkout_path / path + with socket.socket(socket.AF_UNIX) as sock: + # Call fchmod() before we create the socket to ensure that its initial + # permissions are not looser than requested. The OS will still honor the + # umask when creating the socket. + os.fchmod(sock.fileno(), mode) + sock.bind(str(socket_path)) + sock.listen(10) + # Call chmod() update the permissions ignoring the umask. + # Note that we unfortunately must use path.chmod() here rather than + # os.fchmod(): Linux appears to ignore fchmod() calls after the socket has + # already been bound. + socket_path.chmod(mode) + snapshot_types: Dict[str, Type[BaseSnapshot]] = {} @@ -243,9 +475,43 @@ def generate(snapshot_type: Type[T]) -> Iterator[T]: temporary directory that will be cleaned up when exiting the `with` context. """ with create_tmp_dir() as tmpdir: - with snapshot_type(tmpdir) as snapshot: - snapshot.generate() - yield snapshot + snapshot = snapshot_type(tmpdir) + snapshot.generate() + yield snapshot + + +class UnknownSnapshotTypeError(ValueError): + def __init__(self, type_name: str) -> None: + super().__init__(f"unknown snapshot type {type_name!r}") + self.type_name = type_name + + +def unpack_into(snapshot_path: Path, output_path: Path) -> BaseSnapshot: + """Unpack a snapshot into the specified output directory. + + Returns the appropriate BaseSnapshot subclass for this snapshot. + """ + # GNU tar is smart enough to automatically figure out the correct + # decompression method. + untar_cmd = ["tar", "-xf", str(snapshot_path)] + subprocess.check_call(untar_cmd, cwd=output_path) + + data_dir = output_path / "data" + try: + with (data_dir / "info.json").open("r") as info_file: + info = json.load(info_file) + + type_name = info["type"] + snapshot_type = snapshot_types.get(type_name) + if snapshot_type is None: + raise UnknownSnapshotTypeError(type_name) + + snapshot = snapshot_type(output_path) + snapshot.resume() + return snapshot + except Exception as ex: + util.cleanup_tmp_dir(data_dir) + raise def _import_snapshot_modules() -> None: diff --git a/eden/integration/snapshot/test_snapshots.py b/eden/integration/snapshot/test_snapshots.py new file mode 100644 index 0000000000..fee31fcf4a --- /dev/null +++ b/eden/integration/snapshot/test_snapshots.py @@ -0,0 +1,131 @@ +#!/usr/bin/env python3 +# +# Copyright (c) 2016-present, Facebook, Inc. +# All rights reserved. +# +# This source code is licensed under the BSD-style license found in the +# LICENSE file in the root directory of this source tree. An additional grant +# of patent rights can be found in the PATENTS file in the same directory. + +import os +import stat +import unittest +from pathlib import Path +from typing import Callable + +from eden.integration.lib import edenclient + +from . import snapshot as snapshot_mod, verify as verify_mod + + +class Test(unittest.TestCase): + """Tests to verify the contents of various saved snapshots. + + All of the test functions in this class are dynamically added by register_tests() + """ + + def _test_snapshot(self, snapshot_path: Path) -> None: + with snapshot_mod.create_tmp_dir() as tmp_dir: + snapshot = snapshot_mod.unpack_into(snapshot_path, tmp_dir) + self._run_test(snapshot) + + def _run_test(self, snapshot: snapshot_mod.BaseSnapshot) -> None: + verifier = verify_mod.SnapshotVerifier() + snapshot.verify(verifier) + + # Fail the test if any errors were found. + # The individual errors will have been printed out previously + # as they were found. + if verifier.errors: + self.fail(f"found {len(verifier.errors)} errors") + + +class InfraTests(unittest.TestCase): + """Tests for the snapshot generation/verification code itself.""" + + NUM_SNAPSHOTS = 0 + + def test_snapshot_list(self) -> None: + # Ensure that at least one snapshot file was found, so that the tests will + # fail if we somehow can't find the snapshot data directory correctly. + self.assertGreater(self.NUM_SNAPSHOTS, 0) + + def test_verify_directory(self) -> None: + File = verify_mod.ExpectedFile + Socket = verify_mod.ExpectedSocket + Symlink = verify_mod.ExpectedSymlink + + expected = [ + File("a/b/normal.txt", b"abc\n", 0o644), + File("a/b/normal_exe.exe", b"abc\n", 0o755), + File("a/b/missing.txt", b"abc\n", 0o644), + File("a/b/wrong_perms.txt", b"abc\n", 0o644), + File("a/b/wrong_file_type.txt", b"abc\n", 0o644), + Socket("a/normal.sock", 0o644), + Socket("a/exe.sock", 0o755), + Symlink("a/normal.link", b"symlink contents", 0o777), + Symlink("a/missing.link", b"missing symlink", 0o777), + ] + + # Define a subclass of HgSnapshot. We use define this solely so we can use its + # helper write_file(), make_socket(), and mkdir() methods + class MockSnapshot(snapshot_mod.HgSnapshot): + def populate_backing_repo(self) -> None: + pass + + def populate_checkout(self) -> None: + pass + + def verify_snapshot_data( + self, verifier: verify_mod.SnapshotVerifier, eden: edenclient.EdenFS + ) -> None: + pass + + with snapshot_mod.create_tmp_dir() as tmp_dir: + snapshot = MockSnapshot(tmp_dir) + snapshot.data_dir.mkdir() + snapshot.checkout_path.mkdir() + snapshot.write_file("a/b/normal.txt", b"abc\n", 0o644) + snapshot.write_file("a/b/normal_exe.exe", b"abc\n", 0o755) + snapshot.write_file("a/b/wrong_perms.txt", b"abc\n", 0o755) + snapshot.make_socket("a/b/wrong_file_type.txt", 0o755) + snapshot.make_socket("a/normal.sock", 0o644) + snapshot.make_socket("a/exe.sock", 0o755) + os.symlink(b"symlink contents", snapshot.checkout_path / "a/normal.link") + # The verifier code only checks files, not directories, so it should not + # complain about extra directories that may be present. + snapshot.mkdir("a/b/c/extra_dir", 0o755) + + verifier = verify_mod.SnapshotVerifier() + verifier.verify_directory(snapshot.checkout_path, expected) + + expected_errors = [ + "a/b/missing.txt: file not present in snapshot", + "a/missing.link: file not present in snapshot", + f"a/b/wrong_file_type.txt: expected file type to be {stat.S_IFREG:#o}, " + f"found {stat.S_IFSOCK:#o}", + f"a/b/wrong_file_type.txt: expected permissions to be 0o644, found 0o755", + "a/b/wrong_perms.txt: expected permissions to be 0o644, found 0o755", + ] + self.assertEqual(sorted(verifier.errors), sorted(expected_errors)) + + +def register_tests() -> None: + # Create one test function for each snapshot + snapshot_dir = Path("eden/test-data/snapshots").resolve() + for snapshot in snapshot_dir.iterdir(): + # We don't use Path.stem here since it only strips off the very last suffix, + # so foo.tar.bz2 becomes foo.tar rather than foo. + stem = snapshot.name.split(".", 1)[0] + setattr(Test, f"test_{stem}", _create_test_fn(snapshot)) + InfraTests.NUM_SNAPSHOTS += 1 + + +def _create_test_fn(snapshot: Path) -> Callable[[Test], None]: + def test_fn(self: Test) -> None: + self._test_snapshot(snapshot) + + return test_fn + + +register_tests() diff --git a/eden/integration/snapshot/types/basic.py b/eden/integration/snapshot/types/basic.py index 222a78910f..6b3d15f92b 100644 --- a/eden/integration/snapshot/types/basic.py +++ b/eden/integration/snapshot/types/basic.py @@ -7,6 +7,8 @@ # LICENSE file in the root directory of this source tree. An additional grant # of patent rights can be found in the PATENTS file in the same directory. +from eden.integration.lib import edenclient +from eden.integration.snapshot import verify as verify_mod from eden.integration.snapshot.snapshot import HgSnapshot, snapshot_class @@ -15,22 +17,129 @@ from eden.integration.snapshot.snapshot import HgSnapshot, snapshot_class "A simple directory structure with a mix of loaded, materialized, " "and unloaded files.", ) -class BaseSnapshot(HgSnapshot): +class BasicSnapshot(HgSnapshot): def populate_backing_repo(self) -> None: repo = self.backing_repo repo.write_file("README.md", "project docs") - repo.write_file("src/main.c", 'printf("hello world!\\n");\n') - repo.write_file("src/lib.c", "void do_stuff() {}\n") - repo.write_file("src/test/test.c", 'printf("success!\\n");\n') - repo.write_file("include/lib.h", "void do_stuff();\n") - repo.write_file("other/foo.txt", "foo\n") - repo.write_file("other/bar.txt", "bar\n") - repo.write_file("other/a/b/c.txt", "abc\n") + repo.write_file(".gitignore", "ignored.txt\n") + + repo.write_file("main/loaded_dir/loaded_file.c", "loaded") + repo.write_file("main/loaded_dir/not_loaded_file.c", "not loaded") + repo.write_file("main/loaded_dir/not_loaded_exe.sh", "not loaded", mode=0o755) + + repo.write_file( + "main/materialized_subdir/script.sh", "original script contents", mode=0o755 + ) + repo.write_file("main/materialized_subdir/test.c", "original test contents") + repo.write_file("main/materialized_subdir/unmodified.txt", "original contents") + repo.write_file("main/mode_changes/normal_to_exe.txt", "will change mode") + repo.write_file( + "main/mode_changes/exe_to_normal.txt", "will change mode", mode=0o755 + ) + repo.write_file("main/mode_changes/normal_to_readonly.txt", "will be readonly") + + repo.write_file("never_accessed/foo/bar/baz.txt", "baz\n") + repo.write_file("never_accessed/foo/bar/xyz.txt", "xyz\n") + repo.write_file("never_accessed/foo/file.txt", "data\n") repo.commit("Initial commit.") def populate_checkout(self) -> None: - # Load the src directory and the src/lib.c file - self.list_dir("src") - self.read_file("src/lib.c") - # Modify src/test/test.c to force it to be materialized - self.write_file("src/test/test.c", b"new test contents") + # Load the main/loaded_dir directory and the main/loaded_dir/lib.c file + # This currently allocates inode numbers for everything in main/loaded_dir/ and + # causes main/loaded_dir/ to be tracked in the overlay + self.list_dir("main/loaded_dir") + self.read_file("main/loaded_dir/loaded_file.c") + + # Modify some files in main/materialized_subdir to force them to be materialized + self.write_file( + "main/materialized_subdir/script.sh", b"new script contents", 0o755 + ) + self.write_file("main/materialized_subdir/test.c", b"new test contents") + + # Test materializing some files by changing their mode + self.chmod("main/mode_changes/normal_to_exe.txt", 0o755) + self.chmod("main/mode_changes/exe_to_normal.txt", 0o644) + self.chmod("main/mode_changes/normal_to_readonly.txt", 0o400) + + # Create a new top-level directory with some new files + self.write_file("untracked/new/normal.txt", b"new src contents") + self.write_file("untracked/new/normal2.txt", b"extra src contents") + self.write_file("untracked/new/readonly.txt", b"new readonly contents", 0o400) + self.write_file("untracked/executable.exe", b"do stuff", mode=0o755) + self.make_socket("untracked/everybody.sock", mode=0o666) + self.make_socket("untracked/owner_only.sock", mode=0o600) + + # Create some untracked files in an existing tracked directory + self.write_file("main/untracked.txt", b"new new untracked file") + self.write_file("main/ignored.txt", b"new ignored file") + self.write_file("main/untracked_dir/foo.txt", b"foobar") + + def verify_snapshot_data( + self, verifier: verify_mod.SnapshotVerifier, eden: edenclient.EdenFS + ) -> None: + # Confirm that `hg status` reports the correct information + self.verify_hg_status(verifier) + + # Confirm that the files look like what we expect + File = verify_mod.ExpectedFile + Socket = verify_mod.ExpectedSocket + Symlink = verify_mod.ExpectedSymlink + expected_files = [ + # TODO: These symlink permissions should ideally be 0o777 + Symlink(".eden/root", bytes(self.checkout_path), 0o770), + Symlink( + ".eden/client", + bytes(self.eden_state_dir / "clients" / "checkout"), + 0o770, + ), + Symlink(".eden/socket", bytes(self.eden_state_dir / "socket"), 0o770), + File("README.md", b"project docs", 0o644), + File(".gitignore", b"ignored.txt\n", 0o644), + File("main/loaded_dir/loaded_file.c", b"loaded", 0o644), + File("main/loaded_dir/not_loaded_file.c", b"not loaded", 0o644), + File("main/loaded_dir/not_loaded_exe.sh", b"not loaded", 0o755), + File("main/materialized_subdir/script.sh", b"new script contents", 0o755), + File("main/materialized_subdir/test.c", b"new test contents", 0o644), + File( + "main/materialized_subdir/unmodified.txt", b"original contents", 0o644 + ), + File("main/mode_changes/normal_to_exe.txt", b"will change mode", 0o755), + File("main/mode_changes/exe_to_normal.txt", b"will change mode", 0o644), + File( + "main/mode_changes/normal_to_readonly.txt", b"will be readonly", 0o400 + ), + File("main/untracked.txt", b"new new untracked file", 0o644), + File("main/ignored.txt", b"new ignored file", 0o644), + File("main/untracked_dir/foo.txt", b"foobar", 0o644), + File("never_accessed/foo/bar/baz.txt", b"baz\n", 0o644), + File("never_accessed/foo/bar/xyz.txt", b"xyz\n", 0o644), + File("never_accessed/foo/file.txt", b"data\n", 0o644), + File("untracked/new/normal.txt", b"new src contents", 0o644), + File("untracked/new/normal2.txt", b"extra src contents", 0o644), + File("untracked/new/readonly.txt", b"new readonly contents", 0o400), + File("untracked/executable.exe", b"do stuff", 0o755), + Socket("untracked/everybody.sock", 0o666), + Socket("untracked/owner_only.sock", 0o600), + ] + verifier.verify_directory(self.checkout_path, expected_files) + + def verify_hg_status(self, verifier: verify_mod.SnapshotVerifier) -> None: + expected_status = { + "main/materialized_subdir/script.sh": "M", + "main/materialized_subdir/test.c": "M", + "main/mode_changes/normal_to_exe.txt": "M", + "main/mode_changes/exe_to_normal.txt": "M", + # We changed the mode on main/mode_changes/normal_to_readonly.txt, + # but the change isn't significant to mercurial. + "untracked/new/normal.txt": "?", + "untracked/new/normal2.txt": "?", + "untracked/new/readonly.txt": "?", + "untracked/executable.exe": "?", + "untracked/everybody.sock": "?", + "untracked/owner_only.sock": "?", + "main/untracked.txt": "?", + "main/ignored.txt": "I", + "main/untracked_dir/foo.txt": "?", + } + repo = self.hg_repo(self.checkout_path) + verifier.verify_hg_status(repo, expected_status) diff --git a/eden/integration/snapshot/verify.py b/eden/integration/snapshot/verify.py new file mode 100644 index 0000000000..d0f2fb0783 --- /dev/null +++ b/eden/integration/snapshot/verify.py @@ -0,0 +1,174 @@ +#!/usr/bin/env python3 +# +# Copyright (c) 2016-present, Facebook, Inc. +# All rights reserved. +# +# This source code is licensed under the BSD-style license found in the +# LICENSE file in the root directory of this source tree. An additional grant +# of patent rights can be found in the PATENTS file in the same directory. + +import abc +import os +import stat as stat_mod +from pathlib import Path +from typing import Dict, List + +from eden.integration.lib import hgrepo + + +class ExpectedFileBase(metaclass=abc.ABCMeta): + def __init__(self, path: str, perms: int, file_type: int) -> None: + self.path = Path(path) + self.permissions = perms + self.file_type = file_type + + def verify( + self, verifier: "SnapshotVerifier", path: Path, stat_info: os.stat_result + ) -> None: + found_perms = stat_mod.S_IMODE(stat_info.st_mode) + if found_perms != self.permissions: + verifier.error( + f"{self.path}: expected permissions to be {self.permissions:#o}, " + f"found {found_perms:#o}" + ) + found_file_type = stat_mod.S_IFMT(stat_info.st_mode) + if found_file_type != self.file_type: + verifier.error( + f"{self.path}: expected file type to be {self.file_type:#o}, " + f"found {found_file_type:#o}" + ) + else: + self._verify_contents(verifier, path) + + @abc.abstractmethod + def _verify_contents(self, verifier: "SnapshotVerifier", path: Path) -> None: + pass + + def _error(self, msg: str) -> None: + raise ValueError(msg) + + +class ExpectedFile(ExpectedFileBase): + def __init__(self, path: str, contents: bytes, perms: int = 0o644) -> None: + super().__init__(path, perms, stat_mod.S_IFREG) + self.contents = contents + + def _verify_contents(self, verifier: "SnapshotVerifier", path: Path) -> None: + with path.open("rb") as f: + actual_contents = f.read() + if actual_contents != self.contents: + verifier.error( + f"file contents mismatch for {self.path}:\n" + f"expected: {self.contents!r}\n" + f"actual: {actual_contents!r}" + ) + + +class ExpectedSymlink(ExpectedFileBase): + def __init__(self, path: str, contents: bytes, perms: int = 0o777) -> None: + super().__init__(path, perms, stat_mod.S_IFLNK) + self.contents = contents + + def _verify_contents(self, verifier: "SnapshotVerifier", path: Path) -> None: + actual_contents = os.readlink(bytes(path)) + if actual_contents != self.contents: + verifier.error( + f"symlink contents mismatch for {self.path}:\n" + f"expected: {self.contents!r}\n" + f"actual: {actual_contents!r}" + ) + + +class ExpectedSocket(ExpectedFileBase): + def __init__(self, path: str, perms: int = 0o755) -> None: + super().__init__(path, perms, stat_mod.S_IFSOCK) + + def _verify_contents(self, verifier: "SnapshotVerifier", path: Path) -> None: + pass + + +class SnapshotVerifier: + def __init__(self) -> None: + self.errors: List[str] = [] + self.quiet = False + + def error(self, message: str) -> None: + self.errors.append(message) + if not self.quiet: + print(f"==ERROR== {message}") + + def verify_directory(self, path: Path, expected: List[ExpectedFileBase]) -> None: + """Confirm that the contents of a directory match the expected file state.""" + found_files = enumerate_directory(path) + for expected_entry in expected: + file_stat = found_files.pop(expected_entry.path, None) + if file_stat is None: + self.error(f"{expected_entry.path}: file not present in snapshot") + continue + + full_path = path / expected_entry.path + try: + expected_entry.verify(self, full_path, file_stat) + except AssertionError as ex: + self.error(f"{expected_entry.path}: {ex}") + continue + + for path, stat_info in found_files.items(): + if stat_mod.S_ISDIR(stat_info.st_mode): + # Don't require directories to be listed explicitly in the input files + continue + if str(path.parents[0]) == ".hg": + # Don't complain about files inside the .hg directory that the caller + # did not explicitly specify. Mercurial can create a variety of files + # here, and we don't care about checking the exact list of files it + # happened to create when the snapshot was generated. + continue + self.error(f"{path}: unexpected file present in snapshot") + + def verify_hg_status( + self, + repo: hgrepo.HgRepository, + expected: Dict[str, str], + check_ignored: bool = True, + ) -> None: + actual_status = repo.status(include_ignored=check_ignored) + + for path, expected_char in expected.items(): + actual_char = actual_status.pop(path, None) + if expected_char != actual_char: + self.error( + f"{path}: unexpected hg status difference: " + f"reported as {actual_char}, expected {expected_char}" + ) + + for path, actual_char in actual_status.items(): + self.error( + f"{path}: unexpected hg status difference: " + f"reported as {actual_char}, expected None" + ) + + +def enumerate_directory(path: Path) -> Dict[Path, os.stat_result]: + """ + Recursively walk a directory and return a dictionary of all of the files and + directories it contains. + + Returns a dictionary of [path -> os.stat_result] + The returned paths are relative to the input directory. + """ + entries: Dict[Path, os.stat_result] = {} + _enumerate_directory_helper(path, Path(), entries) + return entries + + +def _enumerate_directory_helper( + root_path: Path, rel_path: Path, results: Dict[Path, os.stat_result] +) -> None: + for entry in os.scandir(root_path / rel_path): + # Current versions of typeshed don't know about the follow_symlinks argument, + # so ignore type errors on the next line. + stat_info = entry.stat(follow_symlinks=False) # type: ignore + entry_path = rel_path / entry.name + results[entry_path] = stat_info + if stat_mod.S_ISDIR(stat_info.st_mode): + _enumerate_directory_helper(root_path, entry_path, results) diff --git a/eden/test-data/snapshots/basic-20181030.tar.xz b/eden/test-data/snapshots/basic-20181030.tar.xz new file mode 100644 index 0000000000000000000000000000000000000000..813378b88d1f977d68e81d9688a0c286ff4386e2 GIT binary patch literal 16508 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 literal 0 HcmV?d00001