summaryrefslogtreecommitdiff
path: root/test/cases/librerelease.bats
blob: e936ea88c9060605e11949b825d30a39be57c8d7 (plain)
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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
load ../lib/common

## helpers ##

# Stub server-side `db-update` for SSH localhost to run
StubDbupdate()
{
	cat <<-eot
#!/bin/bash
echo "==> Updating [fake-repo]"
{
		printf '%s\n' "\$DBSCRIPTS_CONFIG"
		readlink -f -- "\$STAGING"
		find "\$STAGING" -printf '%P\n' | LC_COLLATE=C sort
} > ${tmpdir@Q}/log.txt
eot
}

# Stub server-side `pbot-say` for SSH localhost to run
StubPbotsay()
{
	cat <<-eot
#!/bin/bash
echo "\$*" >${tmpdir@Q}/pbot.txt
eot
}

StageFiles()
{
	local staging_dir="$1"

	mkdir -p "$staging_dir"/repo1 "$staging_dir"/repo2/sub
	touch "$staging_dir"/repo1/file1            \
	      "$staging_dir"/repo1/file2            \
	      "$staging_dir/repo2/file with spaces" \
	      "$staging_dir"/repo2/sub/subfolder
}


## setup/teardown ##

setup() {
	common_setup

	local ssh_port

	# Configure and start the SSH server
	install -Dm644 /dev/stdin "$tmpdir/etc/ssh/sshd_config" <<-eot
		AuthorizedKeysCommand /usr/bin/cat ${HOME}/.ssh/id_rsa.pub
		AuthorizedKeysCommandUser ${USER}
		PasswordAuthentication no
		AcceptEnv TMPDIR
		AcceptEnv _HOME GNUPGHOME XDG_CACHE_HOME XDG_CONFIG_HOME
		AcceptEnv _PATH LIBRETOOLS_LIBRARY_PATH _librelib_conf_sh_sysconfdir _librelib_conf_sh_pkgconfdir
		AcceptEnv GPGKEY
		ForceCommand HOME=\$_HOME; PATH=\$_PATH; eval "\$SSH_ORIGINAL_COMMAND"
		eot
	ssh-keygen -A -f "$tmpdir"
	ssh_port=$(./lib/runserver "$tmpdir/sshd.pid" \
		sshd -i \
		     -f "$tmpdir/etc/ssh/sshd_config" \
		     -h "$tmpdir/etc/ssh/ssh_host_ecdsa_key")

	# Configure the SSH client
	mkdir -p -- "$HOME/.ssh"
	ssh-keygen -N '' -f "$HOME/.ssh/id_rsa"
	ssh-keyscan -p "$ssh_port" 127.0.0.1 > "$HOME/.ssh/known_hosts"
	touch -- "$HOME/.ssh/config"

	# Configure libretools
	cat >> "$XDG_CONFIG_HOME/libretools/libretools.conf" <<-eot
		TIER0_HOST=127.0.0.1
		TIER0_PORT=$ssh_port
		TIER0_STAGING=${tmpdir@Q}/srv-staging/
		DBSCRIPTS_CONFIG=/etc/dbscripts/config.local.phony
		HOOKPRERELEASE=:
		HOOKPOSTRELEASE=:
		eot
}

teardown() {
	xargs -a "$tmpdir/sshd.pid" kill --

	common_teardown
}


## tests ##

@test "librerelease displays usage text" {
	rm -rf "$XDG_CONFIG_HOME"
	LC_ALL=C librerelease -h > "$tmpdir/stdout" 2> "$tmpdir/stderr"

	[[ "$(sed 1q "$tmpdir/stdout")" =~ Usage:.* ]]
	empty "$tmpdir/stderr"
}

@test "librerelease lists all files" {
	StageFiles "$tmpdir"/workdir/staging

	LC_ALL=C librerelease -l &> "$tmpdir/list" || { status=$?; cat "$tmpdir/list"; return $status; }

	cat > "$tmpdir/list-correct" <<-eot
		  -> repo1
		     file1
		     file2
		  -> repo2
		     file with spaces
		     sub/subfolder
		eot

	diff -u "$tmpdir/list-correct" "$tmpdir/list"
}

@test "librerelease fails if GPGKEY not set" {
	unset GPGKEY

	StageFiles "$tmpdir"/workdir/staging

	LC_ALL=C librerelease -l > "$tmpdir/stdout" 2> "$tmpdir/stderr" || status=$?

	[[ $status != 0 ]]
	empty "$tmpdir/stdout"
	grep GPGKEY "$tmpdir/stderr"
}

@test "librerelease fails if TIER0_HOST not set" {
	unset TIER0_HOST
	sed -i 's|TIER0_HOST=.*|TIER0_HOST=|' "$XDG_CONFIG_HOME/libretools/libretools.conf"

	StageFiles "$tmpdir"/workdir/staging

	LC_ALL=C librerelease -l > "$tmpdir"/stdout 2> "$tmpdir"/stderr || status=$?

	[[ $status != 0 ]]
	grep "libretools.conf variables may have changed" "$tmpdir"/stdout
	grep TIER0_HOST "$tmpdir"/stderr
}

@test "librerelease fails if DBSCRIPTS_CONFIG is not set" {
	cat >> "$XDG_CONFIG_HOME/libretools/libretools.conf" <<-eot
		DBSCRIPTS_CONFIG=''
		eot

	StageFiles "$tmpdir"/workdir/staging

	LC_ALL=C librerelease -l > "$tmpdir/stdout" 2> "$tmpdir/stderr" || status=$?

	[[ $status != 0 ]]
	grep "libretools.conf variables may have changed" "$tmpdir/stdout"
	grep DBSCRIPTS_CONFIG "$tmpdir/stderr"
}

@test "librerelease cleans local staging" {
	local staging_dir="$tmpdir"/workdir/staging ; StageFiles "$staging_dir" ;

	cat > "$tmpdir"/expected-stdout <<-eot
removed 'repo1/file1'
removed 'repo1/file2'
removed 'repo2/file with spaces'
removed 'repo2/sub/subfolder'
eot

	LC_ALL=C librerelease -c > "$tmpdir"/stdout 2> "$tmpdir"/stderr || status=$?

	[[ $status == 0 ]]
	grep "==> Removing files from local staging directory" "$tmpdir"/stderr
	diff -u "$tmpdir"/expected-stdout "$tmpdir"/stdout
	(( ! $(command ls -A1 "$staging_dir" | wc -l) ))
}

@test "librerelease cleans remote staging" {
	local staging_dir="$tmpdir"/srv-staging ; StageFiles "$staging_dir" ;

	cat > "$tmpdir"/expected-stdout <<-eot
removed '$staging_dir//repo1/file1'
removed '$staging_dir//repo1/file2'
removed directory '$staging_dir//repo1'
removed '$staging_dir//repo2/file with spaces'
removed '$staging_dir//repo2/sub/subfolder'
removed directory '$staging_dir//repo2/sub'
removed directory '$staging_dir//repo2'
eot

	LC_ALL=C librerelease -C > "$tmpdir"/stdout 2> "$tmpdir"/stderr || status=$?

	[[ $status == 0 ]]
	grep "==> Removing files from remote staging directory" "$tmpdir"/stderr
	# `rm -rfv` order depends on the underlying filesystem and so
	# we need to sort this for it to be reproducible.
	LC_ALL=C sort <"$tmpdir"/expected-stdout >"$tmpdir"/expected-stdout.sorted
	LC_ALL=C sort <"$tmpdir"/stdout >"$tmpdir"/stdout.sorted
	diff -u "$tmpdir"/expected-stdout.sorted "$tmpdir"/stdout.sorted
	(( ! $(command ls -A1 "$staging_dir" | wc -l) ))
}

@test "librerelease publishes packages successfully" {
	# Add server-side stub
	install -Dm755 /dev/stdin "$tmpdir/bin/db-update" < <(StubDbupdate) # writes log.txt
	PATH=$tmpdir/bin:$PATH

	# Log which directories the hooks are run in.
	cat >> "$XDG_CONFIG_HOME/libretools/libretools.conf" <<-eot
		HOOKPRERELEASE='pwd > ${tmpdir@Q}/prerelease.txt'
		HOOKPOSTRELEASE='pwd > ${tmpdir@Q}/postrelease.txt'
		eot

	StageFiles "$tmpdir"/workdir/staging

	# Run
	librerelease

	# Make sure everything went OK
	pwd > "$tmpdir/pwd.txt"
	cat > "$tmpdir/log-correct.txt" <<-eot
		/etc/dbscripts/config.local.phony
		$(readlink -f -- "$tmpdir/srv-staging")

		repo1
		repo1/file1
		repo1/file1.sig
		repo1/file2
		repo1/file2.sig
		repo2
		repo2/file with spaces
		repo2/file with spaces.sig
		repo2/sub
		repo2/sub/subfolder
		repo2/sub/subfolder.sig
		eot
	diff -u "$tmpdir/log-correct.txt" "$tmpdir/log.txt"
	diff -u "$tmpdir/pwd.txt" "$tmpdir/prerelease.txt"
	diff -u "$tmpdir/pwd.txt" "$tmpdir/postrelease.txt"
}

@test "librerelease notifies pbot" {
	# Add server-side stubs
	install -Dm755 /dev/stdin "$tmpdir"/bin/db-update < <(StubDbupdate) # writes log.txt
	install -Dm755 /dev/stdin "$tmpdir"/bin/pbot-say  < <(StubPbotsay ) # writes pbot.txt
	PATH=$tmpdir/bin:$PATH

	StageFiles "$tmpdir"/workdir/staging

	LC_ALL=C librerelease 2> "$tmpdir"/stderr

	grep -E "  -> Notifying pbot:" "$tmpdir"/stderr
	#grep "     $USER just published:" "$tmpdir"/pbot.txt
	grep "$USER just published:" "$tmpdir"/pbot.txt
}

@test "librerelease logs-in as TIER0_LOGIN" {
	PATH=$tmpdir/bin:$PATH

	echo "TIER0_LOGIN=tier0-user" >> "$XDG_CONFIG_HOME"/libretools/libretools.conf
	LC_ALL=C librerelease -C > "$tmpdir"/stdout 2> "$tmpdir"/stderr || status=$?

	[[ $status != 0 ]]
	empty "$tmpdir"/stdout
	grep "tier0-user@127.0.0.1: Permission denied" "$tmpdir"/stderr
}