-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathdeploy.sh
executable file
·121 lines (91 loc) · 3.24 KB
/
deploy.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#!@runtimeShell@
set -euo pipefail
export PATH="@path@"
nixos_root="@nixosRoot@"
secrets_root="@secretsRoot@"
ssh_control_path="~/.ssh/master-%r@%h:%p.sock"
usage() {
echo "$0 [-nps] <machine>"
}
machine_toplevel_link() { echo "${1}.system"; }
machine_ssh() {
local machine="${1}"
shift
ssh -oControlMaster=auto \
-oControlPath=\"${ssh_control_path}\" \
-oControlPersist=5m \
"${machine}" -- "${@}"
}
realize_ssh() {
local machine="${1}"
# Instantiate configuration on local machine. This prevents underpowered
# machines from having to perform the evaluation themselves.
toplevel_drv=$(nix eval --raw --show-trace \
"${nixos_root}#nixosConfigurations.${machine}.config.system.build.toplevel.drvPath") || return 1
# Copy instantiated (but not realized config) to machine
nix copy --derivation --to "ssh://${machine}" "${toplevel_drv}" 1>&2
machine_ssh "${machine}" nix-store --realize "${toplevel_drv}" \
--add-root "$(machine_toplevel_link "${machine}")" --indirect >/dev/null || return 1
echo "$(nix-store --query --outputs "${toplevel_drv}")"
}
realize_hydra() {
local machine="${1}"
# Uses a hardcoded netrc file for now because I am probably going to open up
# my Hydra instance soon
local toplevel
toplevel="$(curl -sL --netrc-file /etc/nix/netrc -H 'Accept: application/json' \
"https://hydra.benwolsieffer.com/job/localpkgs/release/machines.${machine}/latest" \
| jq -r .buildoutputs.out.path)" || return 1
machine_ssh "${machine}" nix-store --realize "${toplevel}" \
--add-root "$(machine_toplevel_link "${machine}")" --indirect >/dev/null || return 1
echo "${toplevel}"
}
realize_builder() {
local machine="${1}"
# Instantiate configuration on local machine.
toplevel_drv="$(nix eval --raw --show-trace \
"${nixos_root}#nixosConfigurations.${machine}.config.system.build.toplevel.drvPath")" || return 1
# Copy instantiated (but not realized) config to builder
nix copy --derivation --to "ssh://HP-Z420" "${toplevel_drv}" 1>&2 || return 1
# Build on builder
toplevel="$(machine_ssh HP-Z420 nix-store --realize "${toplevel_drv}")" || return 1
machine_ssh "${machine}" nix-store --realize "${toplevel}" \
--add-root "$(machine_toplevel_link "${machine}")" --indirect >/dev/null || return 1
echo "${toplevel}"
}
activate() {
local machine="${1}"
local toplevel="${2}"
local action="${3}"
machine_ssh "${machine}" \
sudo -Sv 2\>/dev/null \&\& \
sudo -n nix-env -p /nix/var/nix/profiles/system --set "${toplevel}" \&\& \
sudo -n "${toplevel}/bin/switch-to-configuration" "${action}" \&\& \
: \
<<< "${sudo_password}"
}
deploy_hydra=0
use_builder=0 # Whether to run builds locally
action=switch
while getopts "pba:" opt; do
case "$opt" in
p) deploy_hydra=1 ;;
b) use_builder=1 ;;
a) action="${OPTARG}" ;;
\?) usage; exit ;;
esac
done
shift "$(($OPTIND -1))"
nixos-secrets -c "${secrets_root}" check
machine="${1}"
shift
if [ "${deploy_hydra}" -eq 1 ]; then
toplevel="$(realize_hydra "${machine}")"
elif [ "${use_builder}" -eq 1 ]; then
toplevel="$(realize_builder "${machine}")"
else
toplevel="$(realize_ssh "${machine}")"
fi
read -sp "[sudo] password for ${USER}: " sudo_password
echo
activate "${machine}" "${toplevel}" "${action}"