test-lib.sh 9 KB
Newer Older
1 2 3 4 5 6 7 8
# Test framework for go-ipfs
#
# Copyright (c) 2014 Christian Couder
# MIT Licensed; see the LICENSE file in this repository.
#
# We are using sharness (https://github.com/mlafeldt/sharness)
# which was extracted from the Git test framework.

9 10 11
# use the ipfs tool to test against

# add current directory to path, for ipfs tool.
dignifiedquire's avatar
dignifiedquire committed
12 13
BIN=$(cd .. && echo `pwd`/bin)
PATH=${BIN}:${PATH}
14

15 16 17 18 19
# set sharness verbosity. we set the env var directly as
# it's too late to pass in --verbose, and --verbose is harder
# to pass through in some cases.
test "$TEST_VERBOSE" = 1 && verbose=t

20
# assert the `ipfs` we're using is the right one.
dignifiedquire's avatar
dignifiedquire committed
21
if test `which ipfs` != ${BIN}/ipfs; then
22 23 24 25 26
	echo >&2 "Cannot find the tests' local ipfs tool."
	echo >&2 "Please check test and ipfs tool installation."
	exit 1
fi

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
27 28 29 30 31

# source the common hashes first.
. lib/test-lib-hashes.sh


32
SHARNESS_LIB="lib/sharness/sharness.sh"
33

34 35 36 37 38 39 40 41
. "$SHARNESS_LIB" || {
	echo >&2 "Cannot source: $SHARNESS_LIB"
	echo >&2 "Please check Sharness installation."
	exit 1
}

# Please put go-ipfs specific shell functions below

42
# grab + output options
43
test "$TEST_NO_FUSE" != 1 && test_set_prereq FUSE
44
test "$TEST_EXPENSIVE" = 1 && test_set_prereq EXPENSIVE
45
test "$TEST_NO_DOCKER" != 1 && type docker && test_set_prereq DOCKER
46

47 48 49 50 51 52
if test "$TEST_VERBOSE" = 1; then
	echo '# TEST_VERBOSE='"$TEST_VERBOSE"
	echo '# TEST_NO_FUSE='"$TEST_NO_FUSE"
	echo '# TEST_EXPENSIVE='"$TEST_EXPENSIVE"
fi

53 54 55
# source our generic test lib
. ../../ipfs-test-lib.sh

56 57 58
# source iptb lib
. ../lib/iptb-lib.sh

59
test_cmp_repeat_10_sec() {
Christian Couder's avatar
Christian Couder committed
60
	for i in $(test_seq 1 100)
61
	do
62
		test_cmp "$1" "$2" >/dev/null && return
63
		go-sleep 100ms
64 65 66 67
	done
	test_cmp "$1" "$2"
}

68
test_run_repeat_60_sec() {
69
	for i in $(test_seq 1 600)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
70
	do
71 72
		(test_eval_ "$1") && return
		go-sleep 100ms
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
73 74 75 76
	done
	return 1 # failed
}

77
test_wait_output_n_lines_60_sec() {
78
	for i in $(test_seq 1 600)
79
	do
80 81
		test $(cat "$1" | wc -l | tr -d " ") -ge $2 && return
		go-sleep 100ms
82
	done
83
	actual=$(cat "$1" | wc -l | tr -d " ")
84
	test_fsh "expected $2 lines of output. got $actual"
85 86
}

87
test_wait_open_tcp_port_10_sec() {
88 89
	for i in $(test_seq 1 100)
	do
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
90 91 92 93 94
		# this is not a perfect check, but it's portable.
		# cant count on ss. not installed everywhere.
		# cant count on netstat using : or . as port delim. differ across platforms.
		echo $(netstat -aln | egrep "^tcp.*LISTEN" | egrep "[.:]$1" | wc -l) -gt 0
		if [ $(netstat -aln | egrep "^tcp.*LISTEN" | egrep "[.:]$1" | wc -l) -gt 0 ]; then
95 96
			return 0
		fi
97
		go-sleep 100ms
98 99 100 101
	done
	return 1
}

102 103 104 105 106 107

# test_config_set helps us make sure _we really did set_ a config value.
# it sets it and then tests it. This became elaborate because ipfs config
# was setting really weird things and am not sure why.
test_config_set() {

Etienne Laurin's avatar
Etienne Laurin committed
108
	# grab flags (like --bool in "ipfs config --bool")
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
	test_cfg_flags="" # unset in case.
	test "$#" = 3 && { test_cfg_flags=$1; shift; }

	test_cfg_key=$1
	test_cfg_val=$2

	# when verbose, tell the user what config values are being set
	test_cfg_cmd="ipfs config $test_cfg_flags \"$test_cfg_key\" \"$test_cfg_val\""
	test "$TEST_VERBOSE" = 1 && echo "$test_cfg_cmd"

	# ok try setting the config key/val pair.
	ipfs config $test_cfg_flags "$test_cfg_key" "$test_cfg_val"
	echo "$test_cfg_val" >cfg_set_expected
	ipfs config "$test_cfg_key" >cfg_set_actual
	test_cmp cfg_set_expected cfg_set_actual
}

126
test_init_ipfs() {
127

128 129 130 131 132 133 134 135 136 137 138
	# we have a problem where initializing daemons with the same api port
	# often fails-- it hangs indefinitely. The proper solution is to make
	# ipfs pick an unused port for the api on startup, and then use that.
	# Unfortunately, ipfs doesnt yet know how to do this-- the api port
	# must be specified. Until ipfs learns how to do this, we must use
	# specific port numbers, which may still fail but less frequently
	# if we at least use different ones.

	# Using RANDOM like this is clearly wrong-- it samples with replacement
	# and it doesnt even check the port is unused. this is a trivial stop gap
	# until the proper solution is implemented.
139
	RANDOM=$$
140 141 142 143 144
	PORT_API=$((RANDOM % 3000 + 5100))
	ADDR_API="/ip4/127.0.0.1/tcp/$PORT_API"

	PORT_GWAY=$((RANDOM % 3000 + 8100))
	ADDR_GWAY="/ip4/127.0.0.1/tcp/$PORT_GWAY"
145

146 147 148 149 150 151
	PORT_SWARM=$((RANDOM % 3000 + 12000))
	ADDR_SWARM="[
  \"/ip4/0.0.0.0/tcp/$PORT_SWARM\"
]"


152 153 154 155
	# we set the Addresses.API config variable.
	# the cli client knows to use it, so only need to set.
	# todo: in the future, use env?

156
	test_expect_success "ipfs init succeeds" '
157
		export IPFS_PATH="$(pwd)/.ipfs" &&
158
		ipfs init -b=1024 > /dev/null
159 160
	'

161
	test_expect_success "prepare config -- mounting and bootstrap rm" '
162
		mkdir mountdir ipfs ipns &&
163 164
		test_config_set Mounts.IPFS "$(pwd)/ipfs" &&
		test_config_set Mounts.IPNS "$(pwd)/ipns" &&
165
		test_config_set Addresses.API "$ADDR_API" &&
166
		test_config_set Addresses.Gateway "$ADDR_GWAY" &&
167
		test_config_set --json Addresses.Swarm "$ADDR_SWARM" &&
168
		ipfs bootstrap rm --all ||
169
		test_fsh cat "\"$IPFS_PATH/config\""
170 171
	'

172 173
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
174
test_config_ipfs_gateway_readonly() {
175
	ADDR_GWAY=$1
176
	test_expect_success "prepare config -- gateway address" '
177 178
		test "$ADDR_GWAY" != "" &&
		test_config_set "Addresses.Gateway" "$ADDR_GWAY"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
179
	'
180 181 182 183 184 185 186

	# tell the user what's going on if they messed up the call.
	if test "$#" = 0; then
		echo "#			Error: must call with an address, for example:"
		echo '#			test_config_ipfs_gateway_readonly "/ip4/0.0.0.0/tcp/5002"'
		echo '#'
	fi
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
187 188 189
}

test_config_ipfs_gateway_writable() {
190 191 192

	test_config_ipfs_gateway_readonly $1

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
193
	test_expect_success "prepare config -- gateway writable" '
Etienne Laurin's avatar
Etienne Laurin committed
194
		test_config_set --bool Gateway.Writable true ||
195
		test_fsh cat "\"$IPFS_PATH/config\""
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
196 197 198
	'
}

199 200
test_launch_ipfs_daemon() {

201
	args="$@"
202

203
	test_expect_success "'ipfs daemon' succeeds" '
204
		ipfs daemon $args >actual_daemon 2>daemon_err &
205 206
	'

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
207 208
	# we say the daemon is ready when the API server is ready.
	test_expect_success "'ipfs daemon' is ready" '
209
		IPFS_PID=$! &&
Henry's avatar
Henry committed
210 211
		pollEndpoint -ep=/version -host=$ADDR_API -v -tout=1s -tries=60 2>poll_apierr > poll_apiout ||
		test_fsh cat actual_daemon || test_fsh cat daemon_err || test_fsh cat poll_apierr || test_fsh cat poll_apiout
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
212 213 214 215
	'

	if test "$ADDR_GWAY" != ""; then
		test_expect_success "'ipfs daemon' output includes Gateway address" '
Henry's avatar
Henry committed
216 217
			pollEndpoint -ep=/version -host=$ADDR_GWAY -v -tout=1s -tries=60 2>poll_gwerr > poll_gwout ||
			test_fsh cat daemon_err || test_fsh cat poll_gwerr || test_fsh cat poll_gwout
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
218 219
		'
	fi
220 221
}

222 223 224 225 226 227 228 229
do_umount() {
    if [ "$(uname -s)" = "Linux" ]; then
	fusermount -u "$1"
    else
	umount "$1"
    fi
}

230
test_mount_ipfs() {
231

232
	# make sure stuff is unmounted first.
233
	test_expect_success FUSE "'ipfs mount' succeeds" '
234 235
		do_umount "$(pwd)/ipfs" || true &&
		do_umount "$(pwd)/ipns" || true &&
236 237 238 239 240 241 242 243
		ipfs mount >actual
	'

	test_expect_success FUSE "'ipfs mount' output looks good" '
		echo "IPFS mounted at: $(pwd)/ipfs" >expected &&
		echo "IPNS mounted at: $(pwd)/ipns" >>expected &&
		test_cmp expected actual
	'
244 245 246 247 248 249 250 251 252

}

test_launch_ipfs_daemon_and_mount() {

	test_init_ipfs
	test_launch_ipfs_daemon
	test_mount_ipfs

253 254 255
}

test_kill_repeat_10_sec() {
256 257
	# try to shut down once + wait for graceful exit
	kill $1
Christian Couder's avatar
Christian Couder committed
258
	for i in $(test_seq 1 100)
259
	do
260
		go-sleep 100ms
261 262
		! kill -0 $1 2>/dev/null && return
	done
263 264 265

	# if not, try once more, which will skip graceful exit
	kill $1
266
	go-sleep 1s
267 268 269 270 271
	! kill -0 $1 2>/dev/null && return

	# ok, no hope. kill it to prevent it messing with other tests
	kill -9 $1 2>/dev/null
	return 1
272 273
}

274
test_kill_ipfs_daemon() {
275

276
	test_expect_success "'ipfs daemon' is still running" '
277 278 279
		kill -0 $IPFS_PID
	'

280
	test_expect_success "'ipfs daemon' can be killed" '
281
		test_kill_repeat_10_sec $IPFS_PID
282 283
	'
}
284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304

test_curl_resp_http_code() {
	curl -I "$1" >curl_output || {
		echo "curl error with url: '$1'"
		echo "curl output was:"
		cat curl_output
		return 1
	}
	shift &&
	RESP=$(head -1 curl_output) &&
	while test "$#" -gt 0
	do
		expr "$RESP" : "$1" >/dev/null && return
		shift
	done
	echo "curl response didn't match!"
	echo "curl response was: '$RESP'"
	echo "curl output was:"
	cat curl_output
	return 1
}
305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323

test_must_be_empty() {
	if test -s "$1"
	then
		echo "'$1' is not empty, it contains:"
		cat "$1"
		return 1
	fi
}

test_should_contain() {
	test "$#" = 2 || error "bug in the test script: not 2 parameters to test_should_contain"
	if ! grep -q "$1" "$2"
	then
		echo "'$2' does not contain '$1', it contains:"
		cat "$2"
		return 1
	fi
}
Jeromy's avatar
Jeromy committed
324 325 326 327

test_str_contains() {
	find=$1
	shift
328
	echo "$@" | egrep "\b$find\b" >/dev/null
Jeromy's avatar
Jeromy committed
329
}
rht's avatar
rht committed
330 331 332 333 334 335 336 337 338 339

disk_usage() {
    # normalize du across systems
    case $(uname -s) in
        Linux)
            DU="du -sb"
            ;;
        FreeBSD)
            DU="du -s -A -B 1"
            ;;
rht's avatar
rht committed
340 341
        Darwin | DragonFly | *)
            DU="du -s"
rht's avatar
rht committed
342 343 344 345
            ;;
    esac
        $DU "$1" | awk "{print \$1}"
}
346 347 348 349 350 351 352 353 354 355 356 357 358 359

# output a file's permission in human readable format
generic_stat() {
    # normalize stat across systems
    case $(uname -s) in
        Linux)
            _STAT="stat -c %A"
            ;;
        FreeBSD | Darwin | DragonFly)
            _STAT="stat -f %Sp"
            ;;
    esac
    $_STAT "$1"
}
360 361

test_check_peerid() {
362 363 364 365 366
	peeridlen=$(echo "$1" | tr -dC "[:alnum:]" | wc -c | tr -d " ") &&
	test "$peeridlen" = "46" || {
		echo "Bad peerid '$1' with len '$peeridlen'"
		return 1
	}
367
}