73b1b758d5
Closes #334
1090 lines
35 KiB
Python
1090 lines
35 KiB
Python
import os
|
|
import sys
|
|
from typing import Any, Callable, Dict, List, Optional
|
|
|
|
import pytest
|
|
|
|
from tests.conftest import Dotfiles
|
|
|
|
|
|
def test_link_canonicalization(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify links to symlinked destinations are canonical.
|
|
|
|
"Canonical", here, means that dotbot does not create symlinks
|
|
that point to intermediary symlinks.
|
|
"""
|
|
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config([{"link": {"~/.f": {"path": "f"}}}])
|
|
|
|
# Point to the config file in a symlinked dotfiles directory.
|
|
dotfiles_symlink = os.path.join(home, "dotfiles-symlink")
|
|
os.symlink(dotfiles.directory, dotfiles_symlink)
|
|
config_file = os.path.join(dotfiles_symlink, os.path.basename(dotfiles.config_filename))
|
|
run_dotbot("-c", config_file, custom=True)
|
|
|
|
expected = os.path.join(dotfiles.directory, "f")
|
|
actual = os.readlink(os.path.abspath(os.path.expanduser("~/.f")))
|
|
if sys.platform == "win32" and actual.startswith("\\\\?\\"):
|
|
actual = actual[4:]
|
|
assert expected == actual
|
|
|
|
|
|
@pytest.mark.parametrize("dst", ["~/.f", "~/f"])
|
|
@pytest.mark.parametrize("include_force", [True, False])
|
|
def test_link_default_source(
|
|
dst: str,
|
|
include_force: bool, # noqa: FBT001
|
|
home: str,
|
|
dotfiles: Dotfiles,
|
|
run_dotbot: Callable[..., None],
|
|
) -> None:
|
|
"""Verify that default sources are calculated correctly.
|
|
|
|
This test includes verifying files with and without leading periods,
|
|
as well as verifying handling of None dict values.
|
|
"""
|
|
|
|
_ = home
|
|
dotfiles.write("f", "apple")
|
|
config = [
|
|
{
|
|
"link": {
|
|
dst: {"force": False} if include_force else None,
|
|
}
|
|
}
|
|
]
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
with open(os.path.abspath(os.path.expanduser(dst))) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
def test_link_environment_user_expansion_target(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify link expands user in target."""
|
|
|
|
_ = home
|
|
src = "~/f"
|
|
target = "~/g"
|
|
with open(os.path.abspath(os.path.expanduser(src)), "w") as file:
|
|
file.write("apple")
|
|
dotfiles.write_config([{"link": {target: src}}])
|
|
run_dotbot()
|
|
|
|
with open(os.path.abspath(os.path.expanduser(target))) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
def test_link_environment_variable_expansion_source(
|
|
monkeypatch: pytest.MonkeyPatch, home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify link expands environment variables in source."""
|
|
|
|
_ = home
|
|
monkeypatch.setenv("APPLE", "h")
|
|
target = "~/.i"
|
|
src = "$APPLE"
|
|
dotfiles.write("h", "grape")
|
|
dotfiles.write_config([{"link": {target: src}}])
|
|
run_dotbot()
|
|
|
|
with open(os.path.abspath(os.path.expanduser(target))) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_environment_variable_expansion_source_extended(
|
|
monkeypatch: pytest.MonkeyPatch, home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify link expands environment variables in extended config syntax."""
|
|
|
|
_ = home
|
|
monkeypatch.setenv("APPLE", "h")
|
|
target = "~/.i"
|
|
src = "$APPLE"
|
|
dotfiles.write("h", "grape")
|
|
dotfiles.write_config([{"link": {target: {"path": src, "relink": True}}}])
|
|
run_dotbot()
|
|
|
|
with open(os.path.abspath(os.path.expanduser(target))) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_environment_variable_expansion_target(
|
|
monkeypatch: pytest.MonkeyPatch, home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify link expands environment variables in target.
|
|
|
|
If the variable doesn't exist, the "variable" must not be replaced.
|
|
"""
|
|
|
|
monkeypatch.setenv("ORANGE", ".config")
|
|
monkeypatch.setenv("BANANA", "g")
|
|
monkeypatch.delenv("PEAR", raising=False)
|
|
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write("h", "grape")
|
|
|
|
config = [
|
|
{
|
|
"link": {
|
|
"~/${ORANGE}/$BANANA": {
|
|
"path": "f",
|
|
"create": True,
|
|
},
|
|
"~/$PEAR": "h",
|
|
}
|
|
}
|
|
]
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, ".config", "g")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, "$PEAR")) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_environment_variable_unset(
|
|
monkeypatch: pytest.MonkeyPatch, home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify link leaves unset environment variables."""
|
|
|
|
monkeypatch.delenv("ORANGE", raising=False)
|
|
dotfiles.write("$ORANGE", "apple")
|
|
dotfiles.write_config([{"link": {"~/f": "$ORANGE"}}])
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, "f")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
def test_link_force_leaves_when_nonexistent(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify force doesn't erase sources when targets are nonexistent."""
|
|
|
|
os.mkdir(os.path.join(home, "dir"))
|
|
open(os.path.join(home, "file"), "a").close()
|
|
config = [
|
|
{
|
|
"link": {
|
|
"~/dir": {"path": "dir", "force": True},
|
|
"~/file": {"path": "file", "force": True},
|
|
}
|
|
}
|
|
]
|
|
dotfiles.write_config(config)
|
|
with pytest.raises(SystemExit):
|
|
run_dotbot()
|
|
|
|
assert os.path.isdir(os.path.join(home, "dir"))
|
|
assert os.path.isfile(os.path.join(home, "file"))
|
|
|
|
|
|
def test_link_force_overwrite_symlink(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify force overwrites a symlinked directory."""
|
|
|
|
os.mkdir(os.path.join(home, "dir"))
|
|
dotfiles.write("dir/f")
|
|
os.symlink(home, os.path.join(home, ".dir"))
|
|
|
|
config = [{"link": {"~/.dir": {"path": "dir", "force": True}}}]
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
assert os.path.isfile(os.path.join(home, ".dir", "f"))
|
|
|
|
|
|
def test_link_glob_1(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify globbing works."""
|
|
|
|
dotfiles.write("bin/a", "apple")
|
|
dotfiles.write("bin/b", "banana")
|
|
dotfiles.write("bin/c", "cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/bin": "bin/*"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, "bin", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, "bin", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, "bin", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_glob_2(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify globbing works with a trailing slash in the source."""
|
|
|
|
dotfiles.write("bin/a", "apple")
|
|
dotfiles.write("bin/b", "banana")
|
|
dotfiles.write("bin/c", "cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/bin/": "bin/*"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, "bin", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, "bin", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, "bin", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_glob_3(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify globbing works with hidden ("period-prefixed") files."""
|
|
|
|
dotfiles.write("bin/.a", "dot-apple")
|
|
dotfiles.write("bin/.b", "dot-banana")
|
|
dotfiles.write("bin/.c", "dot-cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/bin/": "bin/.*"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, "bin", ".a")) as file:
|
|
assert file.read() == "dot-apple"
|
|
with open(os.path.join(home, "bin", ".b")) as file:
|
|
assert file.read() == "dot-banana"
|
|
with open(os.path.join(home, "bin", ".c")) as file:
|
|
assert file.read() == "dot-cherry"
|
|
|
|
|
|
def test_link_glob_4(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify globbing works at the root of the home and dotfiles directories."""
|
|
|
|
dotfiles.write(".a", "dot-apple")
|
|
dotfiles.write(".b", "dot-banana")
|
|
dotfiles.write(".c", "dot-cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~": {
|
|
"path": ".*",
|
|
"glob": True,
|
|
},
|
|
},
|
|
}
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, ".a")) as file:
|
|
assert file.read() == "dot-apple"
|
|
with open(os.path.join(home, ".b")) as file:
|
|
assert file.read() == "dot-banana"
|
|
with open(os.path.join(home, ".c")) as file:
|
|
assert file.read() == "dot-cherry"
|
|
|
|
|
|
@pytest.mark.parametrize("path", ["foo", "foo/"])
|
|
def test_link_glob_ignore_no_glob_chars(
|
|
path: str, home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify ambiguous link globbing fails."""
|
|
|
|
dotfiles.makedirs("foo")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~/foo/": {
|
|
"path": path,
|
|
"glob": True,
|
|
}
|
|
}
|
|
}
|
|
]
|
|
)
|
|
run_dotbot()
|
|
assert os.path.islink(os.path.join(home, "foo"))
|
|
assert os.path.exists(os.path.join(home, "foo"))
|
|
|
|
|
|
def test_link_glob_exclude_1(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify link globbing with an explicit exclusion."""
|
|
|
|
dotfiles.write("config/foo/a", "apple")
|
|
dotfiles.write("config/bar/b", "banana")
|
|
dotfiles.write("config/bar/c", "cherry")
|
|
dotfiles.write("config/baz/d", "donut")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"defaults": {
|
|
"link": {
|
|
"glob": True,
|
|
"create": True,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"link": {
|
|
"~/.config/": {
|
|
"path": "config/*",
|
|
"exclude": ["config/baz"],
|
|
},
|
|
},
|
|
},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".config", "baz"))
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert os.path.islink(os.path.join(home, ".config", "bar"))
|
|
with open(os.path.join(home, ".config", "foo", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".config", "bar", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, ".config", "bar", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_glob_exclude_2(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify deep link globbing with a globbed exclusion."""
|
|
|
|
dotfiles.write("config/foo/a", "apple")
|
|
dotfiles.write("config/bar/b", "banana")
|
|
dotfiles.write("config/bar/c", "cherry")
|
|
dotfiles.write("config/baz/d", "donut")
|
|
dotfiles.write("config/baz/buzz/e", "egg")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"defaults": {
|
|
"link": {
|
|
"glob": True,
|
|
"create": True,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"link": {
|
|
"~/.config/": {
|
|
"path": "config/*/*",
|
|
"exclude": ["config/baz/*"],
|
|
},
|
|
},
|
|
},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".config", "baz"))
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "bar"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "a"))
|
|
with open(os.path.join(home, ".config", "foo", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".config", "bar", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, ".config", "bar", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_glob_exclude_3(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify deep link globbing with an explicit exclusion."""
|
|
|
|
dotfiles.write("config/foo/a", "apple")
|
|
dotfiles.write("config/bar/b", "banana")
|
|
dotfiles.write("config/bar/c", "cherry")
|
|
dotfiles.write("config/baz/d", "donut")
|
|
dotfiles.write("config/baz/buzz/e", "egg")
|
|
dotfiles.write("config/baz/bizz/g", "grape")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"defaults": {
|
|
"link": {
|
|
"glob": True,
|
|
"create": True,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"link": {
|
|
"~/.config/": {
|
|
"path": "config/*/*",
|
|
"exclude": ["config/baz/buzz"],
|
|
},
|
|
},
|
|
},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".config", "baz", "buzz"))
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "bar"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "baz"))
|
|
assert os.path.islink(os.path.join(home, ".config", "baz", "bizz"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "a"))
|
|
with open(os.path.join(home, ".config", "foo", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".config", "bar", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, ".config", "bar", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
with open(os.path.join(home, ".config", "baz", "d")) as file:
|
|
assert file.read() == "donut"
|
|
with open(os.path.join(home, ".config", "baz", "bizz", "g")) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_glob_exclude_4(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify deep link globbing with multiple globbed exclusions."""
|
|
|
|
dotfiles.write("config/foo/a", "apple")
|
|
dotfiles.write("config/bar/b", "banana")
|
|
dotfiles.write("config/bar/c", "cherry")
|
|
dotfiles.write("config/baz/d", "donut")
|
|
dotfiles.write("config/baz/buzz/e", "egg")
|
|
dotfiles.write("config/baz/bizz/g", "grape")
|
|
dotfiles.write("config/fiz/f", "fig")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"defaults": {
|
|
"link": {
|
|
"glob": True,
|
|
"create": True,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"link": {
|
|
"~/.config/": {
|
|
"path": "config/*/*",
|
|
"exclude": ["config/baz/*", "config/fiz/*"],
|
|
},
|
|
},
|
|
},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".config", "baz"))
|
|
assert not os.path.exists(os.path.join(home, ".config", "fiz"))
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "bar"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "a"))
|
|
with open(os.path.join(home, ".config", "foo", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".config", "bar", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, ".config", "bar", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_glob_multi_star(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify link globbing with deep-nested stars."""
|
|
|
|
dotfiles.write("config/foo/a", "apple")
|
|
dotfiles.write("config/bar/b", "banana")
|
|
dotfiles.write("config/bar/c", "cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/.config/": "config/*/*"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "bar"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "a"))
|
|
with open(os.path.join(home, ".config", "foo", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".config", "bar", "b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, ".config", "bar", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("pattern", "expect_file"),
|
|
[
|
|
("conf/*", lambda fruit: fruit),
|
|
("conf/.*", lambda fruit: "." + fruit),
|
|
("conf/[bc]*", lambda fruit: fruit if fruit[0] in "bc" else None),
|
|
("conf/*e", lambda fruit: fruit if fruit[-1] == "e" else None),
|
|
("conf/??r*", lambda fruit: fruit if fruit[2] == "r" else None),
|
|
],
|
|
)
|
|
def test_link_glob_patterns(
|
|
pattern: str,
|
|
expect_file: Callable[[str], Optional[str]],
|
|
home: str,
|
|
dotfiles: Dotfiles,
|
|
run_dotbot: Callable[..., None],
|
|
) -> None:
|
|
"""Verify link glob pattern matching."""
|
|
|
|
fruits = ["apple", "apricot", "banana", "cherry", "currant", "cantalope"]
|
|
for fruit in fruits:
|
|
dotfiles.write("conf/" + fruit, fruit)
|
|
dotfiles.write("conf/." + fruit, "dot-" + fruit)
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/globtest": pattern}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
for fruit in fruits:
|
|
expected = expect_file(fruit)
|
|
if expected is None:
|
|
assert not os.path.exists(os.path.join(home, "globtest", fruit))
|
|
assert not os.path.exists(os.path.join(home, "globtest", "." + fruit))
|
|
elif "." in expected:
|
|
assert not os.path.islink(os.path.join(home, "globtest", fruit))
|
|
assert os.path.islink(os.path.join(home, "globtest", "." + fruit))
|
|
else: # "." not in expected
|
|
assert os.path.islink(os.path.join(home, "globtest", fruit))
|
|
assert not os.path.islink(os.path.join(home, "globtest", "." + fruit))
|
|
|
|
|
|
def test_link_glob_recursive(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify recursive link globbing and exclusions."""
|
|
|
|
dotfiles.write("config/foo/bar/a", "apple")
|
|
dotfiles.write("config/foo/bar/b", "banana")
|
|
dotfiles.write("config/foo/bar/c", "cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/.config/": {"path": "config/**", "exclude": ["config/**/b"]}}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo", "bar"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "bar", "a"))
|
|
assert not os.path.exists(os.path.join(home, ".config", "foo", "bar", "b"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "bar", "c"))
|
|
with open(os.path.join(home, ".config", "foo", "bar", "a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".config", "foo", "bar", "c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_glob_no_match(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify that a glob with no match doesn't raise an error."""
|
|
|
|
_ = home
|
|
dotfiles.makedirs("foo")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/.config/foo": "foo/*"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
|
|
def test_link_glob_single_match(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify linking works even when glob matches exactly one file."""
|
|
# regression test for https://github.com/anishathalye/dotbot/issues/282
|
|
|
|
dotfiles.write("foo/a", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"glob": True, "create": True}}},
|
|
{"link": {"~/.config/foo": "foo/*"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.islink(os.path.join(home, ".config"))
|
|
assert not os.path.islink(os.path.join(home, ".config", "foo"))
|
|
assert os.path.islink(os.path.join(home, ".config", "foo", "a"))
|
|
with open(os.path.join(home, ".config", "foo", "a")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
"sys.platform == 'win32'",
|
|
reason="These if commands won't run on Windows",
|
|
)
|
|
def test_link_if(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify 'if' directives are checked when linking."""
|
|
|
|
os.mkdir(os.path.join(home, "d"))
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~/.f": {"path": "f", "if": "true"},
|
|
"~/.g": {"path": "f", "if": "false"},
|
|
"~/.h": {"path": "f", "if": "[ -d ~/d ]"},
|
|
"~/.i": {"path": "f", "if": "badcommand"},
|
|
},
|
|
}
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".g"))
|
|
assert not os.path.exists(os.path.join(home, ".i"))
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".h")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
"sys.platform == 'win32'",
|
|
reason="These if commands won't run on Windows.",
|
|
)
|
|
def test_link_if_defaults(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify 'if' directive defaults are checked when linking."""
|
|
|
|
os.mkdir(os.path.join(home, "d"))
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"defaults": {
|
|
"link": {
|
|
"if": "false",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"link": {
|
|
"~/.j": {"path": "f", "if": "true"},
|
|
"~/.k": {"path": "f"}, # default is false
|
|
},
|
|
},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".k"))
|
|
with open(os.path.join(home, ".j")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
"sys.platform != 'win32'",
|
|
reason="These if commands only run on Windows.",
|
|
)
|
|
def test_link_if_windows(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify 'if' directives are checked when linking (Windows only)."""
|
|
|
|
os.mkdir(os.path.join(home, "d"))
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~/.f": {"path": "f", "if": 'cmd /c "exit 0"'},
|
|
"~/.g": {"path": "f", "if": 'cmd /c "exit 1"'},
|
|
"~/.h": {"path": "f", "if": 'cmd /c "dir %USERPROFILE%\\d'},
|
|
"~/.i": {"path": "f", "if": 'cmd /c "badcommand"'},
|
|
},
|
|
}
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".g"))
|
|
assert not os.path.exists(os.path.join(home, ".i"))
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".h")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
"sys.platform != 'win32'",
|
|
reason="These if commands only run on Windows.",
|
|
)
|
|
def test_link_if_defaults_windows(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify 'if' directive defaults are checked when linking (Windows only)."""
|
|
|
|
os.mkdir(os.path.join(home, "d"))
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"defaults": {
|
|
"link": {
|
|
"if": 'cmd /c "exit 1"',
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"link": {
|
|
"~/.j": {"path": "f", "if": 'cmd /c "exit 0"'},
|
|
"~/.k": {"path": "f"}, # default is false
|
|
},
|
|
},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
assert not os.path.exists(os.path.join(home, ".k"))
|
|
with open(os.path.join(home, ".j")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
@pytest.mark.parametrize("ignore_missing", [True, False])
|
|
def test_link_ignore_missing(
|
|
ignore_missing: bool, # noqa: FBT001
|
|
home: str,
|
|
dotfiles: Dotfiles,
|
|
run_dotbot: Callable[..., None],
|
|
) -> None:
|
|
"""Verify link 'ignore_missing' is respected when the target is missing."""
|
|
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~/missing_link": {
|
|
"path": "missing",
|
|
"ignore-missing": ignore_missing,
|
|
},
|
|
},
|
|
}
|
|
]
|
|
)
|
|
|
|
if ignore_missing:
|
|
run_dotbot()
|
|
assert os.path.islink(os.path.join(home, "missing_link"))
|
|
else:
|
|
with pytest.raises(SystemExit):
|
|
run_dotbot()
|
|
|
|
|
|
def test_link_leaves_file(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify relink does not overwrite file."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
with open(os.path.join(home, ".f"), "w") as file:
|
|
file.write("grape")
|
|
dotfiles.write_config([{"link": {"~/.f": "f"}}])
|
|
with pytest.raises(SystemExit):
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
@pytest.mark.parametrize("key", ["canonicalize-path", "canonicalize"])
|
|
def test_link_no_canonicalize(key: str, home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify link canonicalization can be disabled."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config([{"defaults": {"link": {key: False}}}, {"link": {"~/.f": {"path": "f"}}}])
|
|
os.symlink(
|
|
dotfiles.directory,
|
|
os.path.join(home, "dotfiles-symlink"),
|
|
target_is_directory=True,
|
|
)
|
|
run_dotbot(
|
|
"-c",
|
|
os.path.join(home, "dotfiles-symlink", os.path.basename(dotfiles.config_filename)),
|
|
custom=True,
|
|
)
|
|
assert "dotfiles-symlink" in os.readlink(os.path.join(home, ".f"))
|
|
|
|
|
|
def test_link_prefix(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify link prefixes are prepended."""
|
|
|
|
dotfiles.write("conf/a", "apple")
|
|
dotfiles.write("conf/b", "banana")
|
|
dotfiles.write("conf/c", "cherry")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~/": {
|
|
"glob": True,
|
|
"path": "conf/*",
|
|
"prefix": ".",
|
|
},
|
|
},
|
|
}
|
|
]
|
|
)
|
|
run_dotbot()
|
|
with open(os.path.join(home, ".a")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".b")) as file:
|
|
assert file.read() == "banana"
|
|
with open(os.path.join(home, ".c")) as file:
|
|
assert file.read() == "cherry"
|
|
|
|
|
|
def test_link_relative(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Test relative linking works."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write("d/e", "grape")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {
|
|
"~/.f": {
|
|
"path": "f",
|
|
},
|
|
"~/.frel": {
|
|
"path": "f",
|
|
"relative": True,
|
|
},
|
|
"~/nested/.frel": {
|
|
"path": "f",
|
|
"relative": True,
|
|
"create": True,
|
|
},
|
|
"~/.d": {
|
|
"path": "d",
|
|
"relative": True,
|
|
},
|
|
},
|
|
}
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
f = os.readlink(os.path.join(home, ".f"))
|
|
if sys.platform == "win32" and f.startswith("\\\\?\\"):
|
|
f = f[4:]
|
|
assert f == os.path.join(dotfiles.directory, "f")
|
|
|
|
frel = os.readlink(os.path.join(home, ".frel"))
|
|
if sys.platform == "win32" and frel.startswith("\\\\?\\"):
|
|
frel = frel[4:]
|
|
assert frel == os.path.normpath("../../dotfiles/f")
|
|
|
|
nested_frel = os.readlink(os.path.join(home, "nested", ".frel"))
|
|
if sys.platform == "win32" and nested_frel.startswith("\\\\?\\"):
|
|
nested_frel = nested_frel[4:]
|
|
assert nested_frel == os.path.normpath("../../../dotfiles/f")
|
|
|
|
d = os.readlink(os.path.join(home, ".d"))
|
|
if sys.platform == "win32" and d.startswith("\\\\?\\"):
|
|
d = d[4:]
|
|
assert d == os.path.normpath("../../dotfiles/d")
|
|
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".frel")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, "nested", ".frel")) as file:
|
|
assert file.read() == "apple"
|
|
with open(os.path.join(home, ".d", "e")) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_relink_leaves_file(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify relink does not overwrite file."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
with open(os.path.join(home, ".f"), "w") as file:
|
|
file.write("grape")
|
|
dotfiles.write_config([{"link": {"~/.f": {"path": "f", "relink": True}}}])
|
|
with pytest.raises(SystemExit):
|
|
run_dotbot()
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_relink_overwrite_symlink(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify relink overwrites symlinks."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
with open(os.path.join(home, "f"), "w") as file:
|
|
file.write("grape")
|
|
os.symlink(os.path.join(home, "f"), os.path.join(home, ".f"))
|
|
dotfiles.write_config([{"link": {"~/.f": {"path": "f", "relink": True}}}])
|
|
run_dotbot()
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
def test_link_relink_relative_leaves_file(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify relink relative does not incorrectly relink file."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
with open(os.path.join(home, ".f"), "w") as file:
|
|
file.write("grape")
|
|
config = [
|
|
{
|
|
"link": {
|
|
"~/.folder/f": {
|
|
"path": "f",
|
|
"create": True,
|
|
"relative": True,
|
|
},
|
|
},
|
|
}
|
|
]
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
mtime = os.stat(os.path.join(home, ".folder", "f")).st_mtime
|
|
|
|
config[0]["link"]["~/.folder/f"]["relink"] = True
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
new_mtime = os.stat(os.path.join(home, ".folder", "f")).st_mtime
|
|
assert mtime == new_mtime
|
|
|
|
|
|
def test_link_defaults_1(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify that link doesn't overwrite non-dotfiles links by default."""
|
|
|
|
with open(os.path.join(home, "f"), "w") as file:
|
|
file.write("grape")
|
|
os.symlink(os.path.join(home, "f"), os.path.join(home, ".f"))
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{
|
|
"link": {"~/.f": "f"},
|
|
}
|
|
]
|
|
)
|
|
with pytest.raises(SystemExit):
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "grape"
|
|
|
|
|
|
def test_link_defaults_2(home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]) -> None:
|
|
"""Verify that explicit link defaults override the implicit default."""
|
|
|
|
with open(os.path.join(home, "f"), "w") as file:
|
|
file.write("grape")
|
|
os.symlink(os.path.join(home, "f"), os.path.join(home, ".f"))
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(
|
|
[
|
|
{"defaults": {"link": {"relink": True}}},
|
|
{"link": {"~/.f": "f"}},
|
|
]
|
|
)
|
|
run_dotbot()
|
|
|
|
with open(os.path.join(home, ".f")) as file:
|
|
assert file.read() == "apple"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"config",
|
|
[
|
|
pytest.param([{"link": {"~/.f": "f"}}], id="unspecified"),
|
|
pytest.param(
|
|
[{"link": {"~/.f": {"path": "f", "type": "symlink"}}}],
|
|
id="specified",
|
|
),
|
|
pytest.param(
|
|
[
|
|
{"defaults": {"link": {"type": "symlink"}}},
|
|
{"link": {"~/.f": "f"}},
|
|
],
|
|
id="symlink set for all links by default",
|
|
),
|
|
],
|
|
)
|
|
def test_link_type_symlink(
|
|
config: List[Dict[str, Any]], home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify that symlinks are created by default, and when specified."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
assert os.path.islink(os.path.join(home, ".f"))
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"config",
|
|
[
|
|
pytest.param(
|
|
[{"link": {"~/.f": {"path": "f", "type": "hardlink"}}}],
|
|
id="specified",
|
|
),
|
|
pytest.param(
|
|
[
|
|
{"defaults": {"link": {"type": "hardlink"}}},
|
|
{"link": {"~/.f": "f"}},
|
|
],
|
|
id="hardlink set for all links by default",
|
|
),
|
|
],
|
|
)
|
|
def test_link_type_hardlink(
|
|
config: List[Dict[str, Any]], home: str, dotfiles: Dotfiles, run_dotbot: Callable[..., None]
|
|
) -> None:
|
|
"""Verify that hardlinks are created when specified."""
|
|
|
|
dotfiles.write("f", "apple")
|
|
assert os.stat(os.path.join(dotfiles.directory, "f")).st_nlink == 1
|
|
dotfiles.write_config(config)
|
|
run_dotbot()
|
|
|
|
assert not os.path.islink(os.path.join(home, ".f"))
|
|
assert os.stat(os.path.join(dotfiles.directory, "f")).st_nlink == 2
|
|
assert os.stat(os.path.join(home, ".f")).st_nlink == 2
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"config",
|
|
[
|
|
pytest.param(
|
|
[{"defaults": {"link": {"type": "default-bogus"}}, "link": {}}],
|
|
id="default link type not recognized",
|
|
),
|
|
pytest.param(
|
|
[{"link": {"~/.f": {"type": "specified-bogus"}}}],
|
|
id="specified link type not recognized",
|
|
),
|
|
],
|
|
)
|
|
def test_unknown_link_type(
|
|
capsys: pytest.CaptureFixture[str],
|
|
config: List[Dict[str, Any]],
|
|
dotfiles: Dotfiles,
|
|
run_dotbot: Callable[..., None],
|
|
) -> None:
|
|
"""Verify that unknown link types are rejected."""
|
|
|
|
dotfiles.write_config(config)
|
|
with pytest.raises(SystemExit):
|
|
run_dotbot()
|
|
stdout, _ = capsys.readouterr()
|
|
assert "link type is not recognized" in stdout
|