2023-09-09 20:39:45 -04:00
|
|
|
import builtins
|
2022-05-13 11:43:29 -04:00
|
|
|
import ctypes
|
2022-04-14 09:17:18 -04:00
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import sys
|
|
|
|
import tempfile
|
2022-04-30 21:46:09 -04:00
|
|
|
from shutil import rmtree
|
2024-12-28 01:01:05 -05:00
|
|
|
from typing import Any, Callable, Generator, List, Optional
|
|
|
|
from unittest import mock
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
import yaml
|
|
|
|
|
|
|
|
import dotbot.cli
|
|
|
|
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def get_long_path(path: str) -> str:
|
2022-05-13 11:43:29 -04:00
|
|
|
"""Get the long path for a given path."""
|
|
|
|
|
|
|
|
# Do nothing for non-Windows platforms.
|
2024-12-28 01:01:05 -05:00
|
|
|
if sys.platform != "win32":
|
2022-05-13 11:43:29 -04:00
|
|
|
return path
|
|
|
|
|
|
|
|
buffer_size = 1000
|
|
|
|
buffer = ctypes.create_unicode_buffer(buffer_size)
|
|
|
|
get_long_path_name = ctypes.windll.kernel32.GetLongPathNameW
|
|
|
|
get_long_path_name(path, buffer, buffer_size)
|
|
|
|
return buffer.value
|
|
|
|
|
|
|
|
|
2023-09-09 20:39:45 -04:00
|
|
|
# On Linux, tempfile.TemporaryFile() requires unlink access.
|
2022-04-14 09:17:18 -04:00
|
|
|
# This list is updated by a tempfile._mkstemp_inner() wrapper,
|
|
|
|
# and its contents are checked by wrapped functions.
|
2024-12-28 01:01:05 -05:00
|
|
|
allowed_tempfile_internal_unlink_calls: List[str] = []
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def wrap_function(
|
|
|
|
function: Callable[..., Any], function_path: str, arg_index: int, kwarg_key: str, root: str
|
|
|
|
) -> Callable[..., Any]:
|
|
|
|
def wrapper(*args: Any, **kwargs: Any) -> Any:
|
|
|
|
value = kwargs[kwarg_key] if kwarg_key in kwargs else args[arg_index]
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
# Allow tempfile.TemporaryFile's internal unlink calls to work.
|
|
|
|
if value in allowed_tempfile_internal_unlink_calls:
|
|
|
|
return function(*args, **kwargs)
|
|
|
|
|
|
|
|
msg = "The '{0}' argument to {1}() must be an absolute path"
|
|
|
|
msg = msg.format(kwarg_key, function_path)
|
|
|
|
assert value == os.path.abspath(value), msg
|
|
|
|
|
|
|
|
msg = "The '{0}' argument to {1}() must be rooted in {2}"
|
|
|
|
msg = msg.format(kwarg_key, function_path, root)
|
2022-04-30 21:42:36 -04:00
|
|
|
assert value[: len(str(root))] == str(root), msg
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
return function(*args, **kwargs)
|
|
|
|
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def wrap_open(root: str) -> Callable[..., Any]:
|
|
|
|
wrapped = builtins.open
|
2022-04-14 09:17:18 -04:00
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def wrapper(*args: Any, **kwargs: Any) -> Any:
|
|
|
|
value = kwargs["file"] if "file" in kwargs else args[0]
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
mode = "r"
|
|
|
|
if "mode" in kwargs:
|
|
|
|
mode = kwargs["mode"]
|
|
|
|
elif len(args) >= 2:
|
|
|
|
mode = args[1]
|
|
|
|
|
|
|
|
msg = "The 'file' argument to open() must be an absolute path"
|
|
|
|
if value != os.devnull and "w" in mode:
|
|
|
|
assert value == os.path.abspath(value), msg
|
|
|
|
|
|
|
|
msg = "The 'file' argument to open() must be rooted in {0}"
|
|
|
|
msg = msg.format(root)
|
|
|
|
if value != os.devnull and "w" in mode:
|
2022-04-30 21:42:36 -04:00
|
|
|
assert value[: len(str(root))] == str(root), msg
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
return wrapped(*args, **kwargs)
|
|
|
|
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def rmtree_error_handler(_function: Any, path: str, _excinfo: Any) -> None:
|
2022-04-14 09:17:18 -04:00
|
|
|
# Handle read-only files and directories.
|
|
|
|
os.chmod(path, 0o777)
|
|
|
|
if os.path.isdir(path):
|
|
|
|
rmtree(path)
|
|
|
|
else:
|
|
|
|
os.unlink(path)
|
|
|
|
|
|
|
|
|
2022-05-13 11:43:29 -04:00
|
|
|
@pytest.fixture(autouse=True, scope="session")
|
2024-12-28 01:01:05 -05:00
|
|
|
def standardize_tmp() -> None:
|
2022-05-13 11:43:29 -04:00
|
|
|
r"""Standardize the temporary directory path.
|
|
|
|
|
|
|
|
On MacOS, `/var` is a symlink to `/private/var`.
|
|
|
|
This creates issues with link canonicalization and relative link tests,
|
|
|
|
so this fixture rewrites environment variables and Python variables
|
|
|
|
to ensure the tests work the same as on Linux and Windows.
|
|
|
|
|
|
|
|
On Windows in GitHub CI, the temporary directory may be a short path.
|
|
|
|
For example, `C:\Users\RUNNER~1\...` instead of `C:\Users\runneradmin\...`.
|
|
|
|
This causes string-based path comparisons to fail.
|
|
|
|
"""
|
|
|
|
|
|
|
|
tmp = tempfile.gettempdir()
|
|
|
|
# MacOS: `/var` is a symlink.
|
|
|
|
tmp = os.path.abspath(os.path.realpath(tmp))
|
|
|
|
# Windows: The temporary directory may be a short path.
|
2024-12-28 01:01:05 -05:00
|
|
|
if sys.platform == "win32":
|
2022-05-13 11:43:29 -04:00
|
|
|
tmp = get_long_path(tmp)
|
|
|
|
os.environ["TMP"] = tmp
|
|
|
|
os.environ["TEMP"] = tmp
|
|
|
|
os.environ["TMPDIR"] = tmp
|
|
|
|
tempfile.tempdir = tmp
|
|
|
|
|
|
|
|
|
2022-04-14 09:17:18 -04:00
|
|
|
@pytest.fixture(autouse=True)
|
2024-12-28 01:01:05 -05:00
|
|
|
def root(standardize_tmp: None) -> Generator[str, None, None]:
|
|
|
|
_ = standardize_tmp
|
2022-04-14 09:17:18 -04:00
|
|
|
"""Create a temporary directory for the duration of each test."""
|
|
|
|
|
|
|
|
# Reset allowed_tempfile_internal_unlink_calls.
|
2024-12-28 01:01:05 -05:00
|
|
|
global allowed_tempfile_internal_unlink_calls # noqa: PLW0603
|
2022-04-14 09:17:18 -04:00
|
|
|
allowed_tempfile_internal_unlink_calls = []
|
|
|
|
|
|
|
|
# Dotbot changes the current working directory,
|
|
|
|
# so this must be reset at the end of each test.
|
|
|
|
current_working_directory = os.getcwd()
|
|
|
|
|
|
|
|
# Create an isolated temporary directory from which to operate.
|
|
|
|
current_root = tempfile.mkdtemp()
|
|
|
|
|
|
|
|
functions_to_wrap = [
|
|
|
|
(os, "chflags", 0, "path"),
|
|
|
|
(os, "chmod", 0, "path"),
|
|
|
|
(os, "chown", 0, "path"),
|
|
|
|
(os, "copy_file_range", 1, "dst"),
|
|
|
|
(os, "lchflags", 0, "path"),
|
|
|
|
(os, "lchmod", 0, "path"),
|
|
|
|
(os, "link", 1, "dst"),
|
|
|
|
(os, "makedirs", 0, "name"),
|
|
|
|
(os, "mkdir", 0, "path"),
|
|
|
|
(os, "mkfifo", 0, "path"),
|
|
|
|
(os, "mknod", 0, "path"),
|
|
|
|
(os, "remove", 0, "path"),
|
|
|
|
(os, "removedirs", 0, "name"),
|
|
|
|
(os, "removexattr", 0, "path"),
|
|
|
|
(os, "rename", 0, "src"), # Check both
|
|
|
|
(os, "rename", 1, "dst"),
|
|
|
|
(os, "renames", 0, "old"), # Check both
|
|
|
|
(os, "renames", 1, "new"),
|
|
|
|
(os, "replace", 0, "src"), # Check both
|
|
|
|
(os, "replace", 1, "dst"),
|
|
|
|
(os, "rmdir", 0, "path"),
|
|
|
|
(os, "setxattr", 0, "path"),
|
|
|
|
(os, "splice", 1, "dst"),
|
|
|
|
(os, "symlink", 1, "dst"),
|
|
|
|
(os, "truncate", 0, "path"),
|
|
|
|
(os, "unlink", 0, "path"),
|
|
|
|
(os, "utime", 0, "path"),
|
|
|
|
(shutil, "chown", 0, "path"),
|
|
|
|
(shutil, "copy", 1, "dst"),
|
|
|
|
(shutil, "copy2", 1, "dst"),
|
|
|
|
(shutil, "copyfile", 1, "dst"),
|
|
|
|
(shutil, "copymode", 1, "dst"),
|
|
|
|
(shutil, "copystat", 1, "dst"),
|
|
|
|
(shutil, "copytree", 1, "dst"),
|
|
|
|
(shutil, "make_archive", 0, "base_name"),
|
|
|
|
(shutil, "move", 0, "src"), # Check both
|
|
|
|
(shutil, "move", 1, "dst"),
|
|
|
|
(shutil, "rmtree", 0, "path"),
|
|
|
|
(shutil, "unpack_archive", 1, "extract_dir"),
|
|
|
|
]
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
patches: List[Any] = []
|
2022-04-14 09:17:18 -04:00
|
|
|
for module, function_name, arg_index, kwarg_key in functions_to_wrap:
|
|
|
|
# Skip anything that doesn't exist in this version of Python.
|
|
|
|
if not hasattr(module, function_name):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# These values must be passed to a separate function
|
|
|
|
# to ensure the variable closures work correctly.
|
2024-12-28 01:01:05 -05:00
|
|
|
function_path = f"{module.__name__}.{function_name}"
|
2022-04-14 09:17:18 -04:00
|
|
|
function = getattr(module, function_name)
|
2022-04-30 21:42:36 -04:00
|
|
|
wrapped = wrap_function(function, function_path, arg_index, kwarg_key, current_root)
|
2022-04-14 09:17:18 -04:00
|
|
|
patches.append(mock.patch(function_path, wrapped))
|
|
|
|
|
|
|
|
# open() must be separately wrapped.
|
2023-09-09 20:39:45 -04:00
|
|
|
function_path = "builtins.open"
|
2022-04-14 09:17:18 -04:00
|
|
|
wrapped = wrap_open(current_root)
|
|
|
|
patches.append(mock.patch(function_path, wrapped))
|
|
|
|
|
|
|
|
# Block all access to bad functions.
|
|
|
|
if hasattr(os, "chroot"):
|
2024-12-28 01:01:05 -05:00
|
|
|
patches.append(mock.patch("os.chroot", return_value=None))
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
# Patch tempfile._mkstemp_inner() so tempfile.TemporaryFile()
|
|
|
|
# can unlink files immediately.
|
2024-12-28 01:01:05 -05:00
|
|
|
mkstemp_inner = tempfile._mkstemp_inner # type: ignore # noqa: SLF001
|
2022-04-14 09:17:18 -04:00
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def wrap_mkstemp_inner(*args: Any, **kwargs: Any) -> Any:
|
2022-04-14 09:17:18 -04:00
|
|
|
(fd, name) = mkstemp_inner(*args, **kwargs)
|
|
|
|
allowed_tempfile_internal_unlink_calls.append(name)
|
|
|
|
return fd, name
|
|
|
|
|
|
|
|
patches.append(mock.patch("tempfile._mkstemp_inner", wrap_mkstemp_inner))
|
|
|
|
|
|
|
|
[patch.start() for patch in patches]
|
|
|
|
try:
|
|
|
|
yield current_root
|
|
|
|
finally:
|
2024-12-18 09:31:31 -05:00
|
|
|
# Patches must be stopped in reverse order because some patches are nested.
|
|
|
|
# Stopping in the reverse order restores the original function.
|
2024-12-28 01:01:05 -05:00
|
|
|
for patch in reversed(patches):
|
|
|
|
patch.stop()
|
2022-04-14 09:17:18 -04:00
|
|
|
os.chdir(current_working_directory)
|
2023-09-10 11:27:54 -04:00
|
|
|
if sys.version_info >= (3, 12):
|
|
|
|
rmtree(current_root, onexc=rmtree_error_handler)
|
|
|
|
else:
|
|
|
|
rmtree(current_root, onerror=rmtree_error_handler)
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2024-12-28 01:01:05 -05:00
|
|
|
def home(monkeypatch: pytest.MonkeyPatch, root: str) -> str:
|
2022-04-14 09:17:18 -04:00
|
|
|
"""Create a home directory for the duration of the test.
|
|
|
|
|
|
|
|
On *nix, the environment variable "HOME" will be mocked.
|
|
|
|
On Windows, the environment variable "USERPROFILE" will be mocked.
|
|
|
|
"""
|
|
|
|
|
|
|
|
home = os.path.abspath(os.path.join(root, "home/user"))
|
|
|
|
os.makedirs(home)
|
2024-12-28 01:01:05 -05:00
|
|
|
if sys.platform == "win32":
|
2022-04-14 09:17:18 -04:00
|
|
|
monkeypatch.setenv("USERPROFILE", home)
|
|
|
|
else:
|
|
|
|
monkeypatch.setenv("HOME", home)
|
2024-12-28 01:01:05 -05:00
|
|
|
return home
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
|
2023-09-09 20:39:45 -04:00
|
|
|
class Dotfiles:
|
2022-04-14 09:17:18 -04:00
|
|
|
"""Create and manage a dotfiles directory for a test."""
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def __init__(self, root: str):
|
2022-04-14 09:17:18 -04:00
|
|
|
self.root = root
|
|
|
|
self.config = None
|
2024-12-28 01:01:05 -05:00
|
|
|
self._config_filename: Optional[str] = None
|
2022-04-14 09:17:18 -04:00
|
|
|
self.directory = os.path.join(root, "dotfiles")
|
|
|
|
os.mkdir(self.directory)
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def makedirs(self, path: str) -> None:
|
2022-04-14 09:17:18 -04:00
|
|
|
os.makedirs(os.path.abspath(os.path.join(self.directory, path)))
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def write(self, path: str, content: str = "") -> None:
|
2022-04-14 09:17:18 -04:00
|
|
|
path = os.path.abspath(os.path.join(self.directory, path))
|
|
|
|
if not os.path.exists(os.path.dirname(path)):
|
|
|
|
os.makedirs(os.path.dirname(path))
|
|
|
|
with open(path, "w") as file:
|
|
|
|
file.write(content)
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def write_config(self, config: Any, serializer: str = "yaml", path: Optional[str] = None) -> str:
|
2022-04-14 09:17:18 -04:00
|
|
|
"""Write a dotbot config and return the filename."""
|
|
|
|
|
|
|
|
assert serializer in {"json", "yaml"}, "Only json and yaml are supported"
|
|
|
|
if serializer == "yaml":
|
2024-12-28 01:01:05 -05:00
|
|
|
serialize: Callable[[Any], str] = yaml.dump
|
2022-04-14 09:17:18 -04:00
|
|
|
else: # serializer == "json"
|
|
|
|
serialize = json.dumps
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
if path is not None:
|
2022-04-14 09:17:18 -04:00
|
|
|
msg = "The config file path must be an absolute path"
|
|
|
|
assert path == os.path.abspath(path), msg
|
|
|
|
|
|
|
|
msg = "The config file path must be rooted in {0}"
|
|
|
|
msg = msg.format(root)
|
2022-04-30 21:42:36 -04:00
|
|
|
assert path[: len(str(root))] == str(root), msg
|
2022-04-14 09:17:18 -04:00
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
self._config_filename = path
|
2022-04-14 09:17:18 -04:00
|
|
|
else:
|
2024-12-28 01:01:05 -05:00
|
|
|
self._config_filename = os.path.join(self.directory, "install.conf.yaml")
|
2022-04-14 09:17:18 -04:00
|
|
|
self.config = config
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
with open(self._config_filename, "w") as file:
|
2022-04-14 09:17:18 -04:00
|
|
|
file.write(serialize(config))
|
2024-12-28 01:01:05 -05:00
|
|
|
return self._config_filename
|
|
|
|
|
|
|
|
@property
|
|
|
|
def config_filename(self) -> str:
|
|
|
|
assert self._config_filename is not None
|
|
|
|
return self._config_filename
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2024-12-28 01:01:05 -05:00
|
|
|
def dotfiles(root: str) -> Dotfiles:
|
2022-04-14 09:17:18 -04:00
|
|
|
"""Create a dotfiles directory."""
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
return Dotfiles(root)
|
2022-04-14 09:17:18 -04:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2024-12-28 01:01:05 -05:00
|
|
|
def run_dotbot(dotfiles: Dotfiles) -> Callable[..., None]:
|
2022-04-14 09:17:18 -04:00
|
|
|
"""Run dotbot.
|
|
|
|
|
|
|
|
When calling `runner()`, only CLI arguments need to be specified.
|
|
|
|
|
|
|
|
If the keyword-only argument *custom* is True
|
|
|
|
then the CLI arguments will not be modified,
|
|
|
|
and the caller will be responsible for all CLI arguments.
|
|
|
|
"""
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
def runner(*argv: Any, **kwargs: Any) -> None:
|
|
|
|
argv = ("dotbot", *argv)
|
2022-04-14 09:17:18 -04:00
|
|
|
if kwargs.get("custom", False) is not True:
|
2024-12-28 01:01:05 -05:00
|
|
|
argv = (*argv, "-c", dotfiles.config_filename)
|
|
|
|
with mock.patch("sys.argv", list(argv)):
|
2022-04-14 09:17:18 -04:00
|
|
|
dotbot.cli.main()
|
|
|
|
|
2024-12-28 01:01:05 -05:00
|
|
|
return runner
|