a291c8690a
GitOrigin-RevId: e6e19f3d81a982a62e1bba08f0b4f7fdc21b4ea0
203 lines
5.4 KiB
Nix
203 lines
5.4 KiB
Nix
{ depot, pkgs, lib, ... }:
|
||
|
||
let
|
||
bins = depot.nix.getBins pkgs.stow [ "stow" ]
|
||
// depot.nix.getBins pkgs.coreutils [ "mkdir" "ln" "printenv" "rm" ]
|
||
// depot.nix.getBins pkgs.xe [ "xe" ]
|
||
// depot.nix.getBins pkgs.lr [ "lr" ]
|
||
// depot.nix.getBins pkgs.nix [ "nix-store" ]
|
||
;
|
||
|
||
# run stow to populate the target directory with the given stow package, read from stowDir.
|
||
# Bear in mind that `stowDirOriginPath` should always be semantically bound to the given `stowDir`, otherwise stow might become rather confused.
|
||
runStow =
|
||
{
|
||
# “stow package” to stow (see manpage)
|
||
# TODO: allow this function to un-stow multiple packages!
|
||
stowPackage
|
||
, # “target directory” to stow in (see manpage)
|
||
targetDir
|
||
, # The “stow directory” (see manpage), containing “stow packages” (see manpage)
|
||
stowDir
|
||
, # representative directory for the stowDir in the file system, against which stow will create relative links.
|
||
# ATTN: this is always overwritten with the contents of `stowDir`! You shouldn’t re-use the same `stowDirOriginPath` for different `stowDir`s, otherwise there might be surprises.
|
||
stowDirOriginPath
|
||
,
|
||
}: depot.nix.writeExecline "stow-${stowPackage}" { } [
|
||
# first, create a temporary stow directory to use as source
|
||
# (stow will use it to determine the origin of files)
|
||
"if"
|
||
[ bins.mkdir "-p" stowDirOriginPath ]
|
||
# remove old symlinks
|
||
"if"
|
||
[
|
||
"pipeline"
|
||
[
|
||
bins.lr
|
||
"-0"
|
||
"-t"
|
||
"depth == 1 && type == l"
|
||
stowDirOriginPath
|
||
]
|
||
bins.xe
|
||
"-0"
|
||
bins.rm
|
||
]
|
||
# create an indirect gc root so our config is not cleaned under our asses by a garbage collect
|
||
"if"
|
||
[
|
||
bins.nix-store
|
||
"--realise"
|
||
"--indirect"
|
||
"--add-root"
|
||
"${stowDirOriginPath}/.nix-stowdir-gc-root"
|
||
stowDir
|
||
]
|
||
# populate with new stow targets
|
||
"if"
|
||
[
|
||
"elglob"
|
||
"-w0"
|
||
"stowPackages"
|
||
"${stowDir}/*"
|
||
bins.ln
|
||
"--force"
|
||
"-st"
|
||
stowDirOriginPath
|
||
"$stowPackages"
|
||
]
|
||
# stow always looks for $HOME/.stowrc to read more arguments
|
||
"export"
|
||
"HOME"
|
||
"/homeless-shelter"
|
||
bins.stow
|
||
# always run restow for now; this does more stat but will remove stale links
|
||
"--restow"
|
||
"--dir"
|
||
stowDirOriginPath
|
||
"--target"
|
||
targetDir
|
||
stowPackage
|
||
];
|
||
|
||
# create a stow dir from a list of drv paths and a stow package name.
|
||
makeStowDir =
|
||
(with depot.nix.yants;
|
||
defun
|
||
[
|
||
(list (struct {
|
||
originalDir = drv;
|
||
stowPackage = string;
|
||
}))
|
||
drv
|
||
])
|
||
(dirs:
|
||
depot.nix.runExecline "make-stow-dir"
|
||
{
|
||
stdin = lib.pipe dirs [
|
||
(map depot.users.Profpatsch.netencode.gen.dwim)
|
||
depot.users.Profpatsch.netstring.toNetstringList
|
||
];
|
||
} [
|
||
"importas"
|
||
"out"
|
||
"out"
|
||
"if"
|
||
[ bins.mkdir "-p" "$out" ]
|
||
"forstdin"
|
||
"-d"
|
||
""
|
||
"-o"
|
||
"0"
|
||
"line"
|
||
"pipeline"
|
||
[
|
||
depot.users.Profpatsch.execline.print-one-env
|
||
"line"
|
||
]
|
||
depot.users.Profpatsch.netencode.record-splice-env
|
||
"importas"
|
||
"-ui"
|
||
"originalDir"
|
||
"originalDir"
|
||
"importas"
|
||
"-ui"
|
||
"stowPackage"
|
||
"stowPackage"
|
||
bins.ln
|
||
"-sT"
|
||
"$originalDir"
|
||
"\${out}/\${stowPackage}"
|
||
]);
|
||
|
||
# this is a dumb way of generating a pure list of packages from a depot namespace.
|
||
readTreeNamespaceDrvs = namespace:
|
||
lib.pipe namespace [
|
||
(lib.filterAttrs (_: v: lib.isDerivation v))
|
||
(lib.mapAttrsToList (k: v: {
|
||
name = k;
|
||
drv = v;
|
||
}))
|
||
];
|
||
|
||
scriptsStow =
|
||
lib.pipe { } [
|
||
(_: makeStowDir [{
|
||
stowPackage = "scripts";
|
||
originalDir = pkgs.linkFarm "scripts-farm"
|
||
([
|
||
{
|
||
name = "scripts/ytextr";
|
||
path = depot.users.Profpatsch.ytextr;
|
||
}
|
||
]
|
||
++
|
||
(lib.pipe depot.users.Profpatsch.aliases [
|
||
readTreeNamespaceDrvs
|
||
(map ({ name, drv }: {
|
||
name = "scripts/${name}";
|
||
path = drv;
|
||
}))
|
||
]));
|
||
}])
|
||
(d: runStow {
|
||
stowDir = d;
|
||
stowPackage = "scripts";
|
||
targetDir = "/home/philip";
|
||
stowDirOriginPath = "/home/philip/.local/share/nix-home/stow-origin";
|
||
})
|
||
];
|
||
|
||
|
||
|
||
terminalEmulatorStow =
|
||
lib.pipe { } [
|
||
(_: makeStowDir [{
|
||
stowPackage = "terminal-emulator";
|
||
originalDir = pkgs.linkFarm "terminal-emulator-farm"
|
||
([
|
||
{
|
||
name = "bin/terminal-emulator";
|
||
path = depot.users.Profpatsch.alacritty;
|
||
}
|
||
]);
|
||
|
||
}])
|
||
(d: runStow {
|
||
stowDir = d;
|
||
stowPackage = "terminal-emulator";
|
||
targetDir = "/home/philip";
|
||
# TODO: this should only be done once, in a single runStow instead of multiple
|
||
stowDirOriginPath = "/home/philip/.local/share/nix-home/stow-origin-terminal-emulator";
|
||
})
|
||
];
|
||
|
||
in
|
||
|
||
# TODO: run multiple stows with runStow?
|
||
# TODO: temp setup
|
||
depot.nix.writeExecline "nix-home" { } [
|
||
"if"
|
||
[ scriptsStow ]
|
||
terminalEmulatorStow
|
||
]
|