Home | History | Annotate | Line # | Download | only in sh
t_option.sh revision 1.2
      1 # $NetBSD: t_option.sh,v 1.2 2016/03/01 12:39:35 christos Exp $
      2 #
      3 # Copyright (c) 2016 The NetBSD Foundation, Inc.
      4 # All rights reserved.
      5 #
      6 # Redistribution and use in source and binary forms, with or without
      7 # modification, are permitted provided that the following conditions
      8 # are met:
      9 # 1. Redistributions of source code must retain the above copyright
     10 #    notice, this list of conditions and the following disclaimer.
     11 # 2. Redistributions in binary form must reproduce the above copyright
     12 #    notice, this list of conditions and the following disclaimer in the
     13 #    documentation and/or other materials provided with the distribution.
     14 #
     15 # THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     16 # ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     17 # TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     18 # PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     19 # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     20 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     21 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     22 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     23 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     24 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     25 # POSSIBILITY OF SUCH DAMAGE.
     26 #
     27 # the implementation of "sh" to test
     28 : ${TEST_SH:="/bin/sh"}
     29 
     30 # The standard
     31 # http://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html
     32 # says:
     33 #	...[lots]
     34 
     35 test_option_on_off()
     36 {
     37 	atf_require_prog tr
     38 
     39 	for opt
     40 	do
     41 				# t is needed, as inside $()` $- appears to lose
     42 				# the 'e' option if it happened to already be
     43 				# set.  Must check if that is what should
     44 				# happen, but that is a different issue.
     45 
     46 		test -z "${opt}" && continue
     47 
     48 		# if we are playing with more that one option at a
     49 		# time, the code below requires that we start with no
     50 		# options set, or it will mis-diagnose the situation
     51 		CLEAR=''
     52 		test "${#opt}" -gt 1 &&
     53   CLEAR='xx="$-" && xx=$(echo "$xx" | tr -d cs) && test -n "$xx" && set +"$xx";'
     54 
     55 		atf_check -s exit:0 -o empty -e empty ${TEST_SH} -c \
     56 			"opt=${opt}"'
     57 			x() {
     58 				echo "ERROR: Unable to $1 option $2" >&2
     59 				exit 1
     60 			}
     61 			s() {
     62 				set -"$1"
     63 				t="$-"
     64 				x=$(echo "$t" | tr -d "$1")
     65 				test "$t" = "$x" && x set "$1"
     66 				return 0
     67 			}
     68 			c() {
     69 				set +"$1"
     70 				t="$-"
     71 				x=$(echo "$t" | tr -d "$1")
     72 				test "$t" != "$x" && x clear "$1"
     73 				return 0
     74 			}
     75 			'"${CLEAR}"'
     76 
     77 			# if we do not do this, -x tracing splatters stderr
     78 			# for some shells, -v does as well (is that correct?)
     79 			case "${opt}" in
     80 			(*[xv]*)	exec 2>/dev/null;;
     81 			esac
     82 
     83 			o="$-"
     84 			x=$(echo "$o" | tr -d "$opt")
     85 
     86 			if [ "$o" = "$x" ]; then	# option was off
     87 				s "${opt}"
     88 				c "${opt}"
     89 			else
     90 				c "${opt}"
     91 				s "${opt}"
     92 			fi
     93 		'
     94 	done
     95 }
     96 
     97 test_optional_on_off()
     98 {
     99 	RET=0
    100 	OPTS=
    101 	for opt
    102 	do
    103 		test "${opt}" = n && continue
    104 		${TEST_SH} -c "set -${opt}" 2>/dev/null  &&
    105 			OPTS="${OPTS} ${opt}" || RET=1
    106 	done
    107 
    108 	test -n "${OPTS}" && test_option_on_off ${OPTS}
    109 
    110 	return "${RET}"
    111 }
    112 
    113 atf_test_case set_a
    114 set_a_head() {
    115 	atf_set "descr" "Tests that 'set -a' turns on all var export " \
    116 	                "and that it behaves as defined by the standard"
    117 }
    118 set_a_body() {
    119 	atf_require_prog env
    120 	atf_require_prog grep
    121 
    122 	test_option_on_off a
    123 
    124 	# without -a, new variables should not be exported (so grep "fails")
    125 	atf_check -s exit:1 -o empty -e empty ${TEST_SH} -ce \
    126 		'unset VAR; set +a; VAR=value; env | grep "^VAR="'
    127 
    128 	# with -a, they should be
    129 	atf_check -s exit:0 -o match:VAR=value -e empty ${TEST_SH} -ce \
    130 		'unset VAR; set -a; VAR=value; env | grep "^VAR="'
    131 }
    132 
    133 atf_test_case set_C
    134 set_C_head() {
    135 	atf_set "descr" "Tests that 'set -C' turns on no clobber mode " \
    136 	                "and that it behaves as defined by the standard"
    137 }
    138 set_C_body() {
    139 	atf_require_prog ls
    140 
    141 	test_option_on_off C
    142 
    143 	# Check that the environment to use for the tests is sane ...
    144 	# we assume current dir is a new tempory directory & is empty
    145 
    146 	test -z "$(ls)" || atf_skip "Test execution directory not clean"
    147 	test -c "/dev/null" || atf_skip "Problem with /dev/null"
    148 
    149 	echo Dummy_Content > Junk_File
    150 	echo Precious_Content > Important_File
    151 
    152 	# Check that we can redirect onto file when -C is not set
    153 	atf_check -s exit:0 -o empty -e empty ${TEST_SH} -c \
    154 		'
    155 		D=$(ls -l Junk_File) || exit 1
    156 		set +C
    157 		echo "Overwrite it now" > Junk_File
    158 		A=$(ls -l Junk_File) || exit 1
    159 		test "${A}" != "${D}"
    160 		'
    161 
    162 	# Check that we cannot redirect onto file when -C is set
    163 	atf_check -s exit:0 -o empty -e not-empty ${TEST_SH} -c \
    164 		'
    165 		D=$(ls -l Important_File) || exit 1
    166 		set -C
    167 		echo "Fail to Overwrite it now" > Important_File
    168 		A=$(ls -l Important_File) || exit 1
    169 		test "${A}" = "${D}"
    170 		'
    171 
    172 	# Check that we can append to file, even when -C is set
    173 	atf_check -s exit:0 -o empty -e empty ${TEST_SH} -c \
    174 		'
    175 		D=$(ls -l Junk_File) || exit 1
    176 		set -C
    177 		echo "Append to it now" >> Junk_File
    178 		A=$(ls -l Junk_File) || exit 1
    179 		test "${A}" != "${D}"
    180 		'
    181 
    182 	# Check that we abort on attempt to redirect onto file when -Ce is set
    183 	atf_check -s not-exit:0 -o empty -e not-empty ${TEST_SH} -c \
    184 		'
    185 		set -Ce
    186 		echo "Fail to Overwrite it now" > Important_File
    187 		echo "Should not reach this point"
    188 		'
    189 
    190 	# Last check that we can override -C for when we really need to
    191 	atf_check -s exit:0 -o empty -e empty ${TEST_SH} -c \
    192 		'
    193 		D=$(ls -l Junk_File) || exit 1
    194 		set -C
    195 		echo "Change the poor bugger again" >| Junk_File
    196 		A=$(ls -l Junk_File) || exit 1
    197 		test "${A}" != "${D}"
    198 		'
    199 }
    200 
    201 atf_test_case set_e
    202 set_e_head() {
    203 	atf_set "descr" "Tests that 'set -e' turns on error detection " \
    204 		"and that a simple case behaves as defined by the standard"
    205 }
    206 set_e_body() {
    207 	test_option_on_off e
    208 
    209 	# Check that -e does nothing if no commands fail
    210 	atf_check -s exit:0 -o match:I_am_OK -e empty \
    211 	    ${TEST_SH} -c \
    212 		'false; printf "%s" I_am; set -e; true; printf "%s\n" _OK'
    213 
    214 	# and that it (silently, but with exit status) aborts if cmd fails
    215 	atf_check -s not-exit:0 -o match:I_am -o not-match:Broken -e empty \
    216 	    ${TEST_SH} -c \
    217 		'false; printf "%s" I_am; set -e; false; printf "%s\n" _Broken'
    218 
    219 	# same, except -e this time is on from the beginning
    220 	atf_check -s not-exit:0 -o match:I_am -o not-match:Broken -e empty \
    221 	    ${TEST_SH} -ec 'printf "%s" I_am; false; printf "%s\n" _Broken'
    222 
    223 	# More checking of -e in other places, there is lots to deal with.
    224 }
    225 
    226 atf_test_case set_f
    227 set_f_head() {
    228 	atf_set "descr" "Tests that 'set -f' turns off pathname expansion " \
    229 	                "and that it behaves as defined by the standard"
    230 }
    231 set_f_body() {
    232 	atf_require_prog ls
    233 
    234 	test_option_on_off f
    235 
    236 	# Check that the environment to use for the tests is sane ...
    237 	# we assume current dir is a new tempory directory & is empty
    238 
    239 	test -z "$(ls)" || atf_skip "Test execution directory not clean"
    240 
    241 	# we will assume that atf will clean up this junk directory
    242 	# when we are done.   But for testing pathname expansion
    243 	# we need files
    244 
    245 	for f in a b c d e f aa ab ac ad ae aaa aab aac aad aba abc bbb ccc
    246 	do
    247 		echo "$f" > "$f"
    248 	done
    249 
    250 	atf_check -s exit:0 -o empty -e empty ${TEST_SH} -ec \
    251 	    'X=$(echo b*); Y=$(echo b*); test "${X}" != "a*";
    252 		test "${X}" = "${Y}"'
    253 
    254 	# now test expansion is different when -f is set
    255 	atf_check -s exit:0 -o empty -e empty ${TEST_SH} -ec \
    256 	   'X=$(echo b*); Y=$(set -f; echo b*); test "${X}" != "${Y}"'
    257 }
    258 
    259 atf_test_case set_n
    260 set_n_head() {
    261 	atf_set "descr" "Tests that 'set -n' supresses command execution " \
    262 	                "and that it behaves as defined by the standard"
    263 }
    264 set_n_body() {
    265 	# pointless to test this, if it turns on, it stays on...
    266 	# test_option_on_off n
    267 	# so just allow the tests below to verify it can be turned on
    268 
    269 	# nothing should be executed, hence no output...
    270 	atf_check -s exit:0 -o empty -e empty \
    271 		${TEST_SH} -enc 'echo ABANDON HOPE; echo ALL YE; echo ...'
    272 
    273 	# this is true even when the "commands" do not exist
    274 	atf_check -s exit:0 -o empty -e empty \
    275 		${TEST_SH} -enc 'ERR; FAIL; ABANDON HOPE'
    276 
    277 	# but if there is a syntax error, it should be detected (w or w/o -e)
    278 	atf_check -s not-exit:0 -o empty -e not-empty \
    279 		${TEST_SH} -enc 'echo JUMP; for frogs swim; echo in puddles'
    280 	atf_check -s not-exit:0 -o empty -e not-empty \
    281 		${TEST_SH} -nc 'echo ABANDON HOPE; echo "ALL YE; echo ...'
    282 	atf_check -s not-exit:0 -o empty -e not-empty \
    283 		${TEST_SH} -enc 'echo ABANDON HOPE;; echo ALL YE; echo ...'
    284 	atf_check -s not-exit:0 -o empty -e not-empty \
    285 		${TEST_SH} -nc 'do YOU ABANDON HOPE; for all eternity?'
    286 
    287 	# now test enabling -n in the middle of a script
    288 	# note that once turned on, it cannot be turned off again.
    289 	#
    290 	# omit more complex cases, as those can send some shells
    291 	# into infinite loops, and believe it or not, that might be OK!
    292 
    293 	atf_check -s exit:0 -o match:first -o not-match:second -e empty \
    294 		${TEST_SH} -c 'echo first; set -n; echo second'
    295 	atf_check -s exit:0 -o match:first -o not-match:third -e empty \
    296 	    ${TEST_SH} -c 'echo first; set -n; echo second; set +n; echo third'
    297 	atf_check -s exit:0 -o inline:'a\nb\n' -e empty \
    298 	    ${TEST_SH} -c 'for x in a b c d
    299 			   do
    300 				case "$x" in
    301 				     a);; b);; c) set -n;; d);;
    302 				esac
    303 				printf "%s\n" "$x"
    304 			   done'
    305 
    306 	# This last one is a bit more complex to explain, so I will not try
    307 
    308 	# First, we need to know what signal number is used for SIGUSR1 on
    309 	# the local (testing) system (signal number is $(( $XIT - 128 )) )
    310 
    311 	# this will take slightly over 1 second elapsed time (the sleep 1)
    312 	# The "10" for the first sleep just needs to be something big enough
    313 	# that the rest of the commands have time to complete, even on
    314 	# very slow testing systems.  10 should be enough.  Otherwise irrelevant
    315 
    316 	# The shell will usually blather to stderr about the sleep 10 being
    317 	# killed, but it affects nothing, so just allow it to cry.
    318 
    319 	(sleep 10 & sleep 1; kill -USR1 $!; wait $!)
    320 	XIT="$?"
    321 
    322 	# The exit value should be an integer > 128 and < 256 (often 158)
    323 	# If it is not just skip the test
    324 
    325 	# If we do run the test, it should take (slightly over) either 1 or 2
    326 	# seconds to complete, depending upon the shell being tested.
    327 
    328 	case "${XIT}" in
    329 	( 129 | 1[3-9][0-9] | 2[0-4][0-9] | 25[0-5] )
    330 
    331 		# The script below should exit with the same code - no output
    332 
    333 		# Or that is the result that seems best explanable.
    334 		# "set -n" in uses like this is not exactly well defined...
    335 
    336 		# This script comes from a member of the austin group
    337 		# (they author changes to the posix shell spec - and more.)
    338 		# The author is also an (occasional?) NetBSD user.
    339 		atf_check -s exit:${XIT} -o empty -e empty ${TEST_SH} -c '
    340 			trap "set -n" USR1
    341 			{ sleep 1; kill -USR1 $$; sleep 1; } &
    342 			false
    343 			wait && echo t || echo f
    344 			wait
    345 			echo foo
    346 		'
    347 		;;
    348 	esac
    349 }
    350 
    351 atf_test_case set_u
    352 set_u_head() {
    353 	atf_set "descr" "Tests that 'set -u' turns on unset var detection " \
    354 	                "and that it behaves as defined by the standard"
    355 }
    356 set_u_body() {
    357 	test_option_on_off u
    358 
    359 	# first make sure it is OK to unset an unset variable
    360 	atf_check -s exit:0 -o match:OK -e empty ${TEST_SH} -ce \
    361 		'unset _UNSET_VARIABLE_; echo OK'
    362 	# even if -u is set
    363 	atf_check -s exit:0 -o match:OK -e empty ${TEST_SH} -cue \
    364 		'unset _UNSET_VARIABLE_; echo OK'
    365 
    366 	# and that without -u accessing an unset variable is harmless
    367 	atf_check -s exit:0 -o match:OK -e empty ${TEST_SH} -ce \
    368 		'unset X; echo ${X}; echo OK'
    369 	# and that the unset variable test expansion works properly
    370 	atf_check -s exit:0 -o match:OKOK -e empty ${TEST_SH} -ce \
    371 		'unset X; printf "%s" ${X-OK}; echo OK'
    372 
    373 	# Next test that with -u set, the shell aborts on access to unset var
    374 	# do not use -e, want to make sure it is -u that causes abort
    375 	atf_check -s not-exit:0 -o not-match:ERR -e not-empty ${TEST_SH} -c \
    376 		'unset X; set -u; echo ${X}; echo ERR'
    377 	# quoting should make no difference...
    378 	atf_check -s not-exit:0 -o not-match:ERR -e not-empty ${TEST_SH} -c \
    379 		'unset X; set -u; echo "${X}"; echo ERR'
    380 
    381 	# Now a bunch of accesses to unset vars, with -u, in ways that are OK
    382 	atf_check -s exit:0 -o match:OK -e empty ${TEST_SH} -ce \
    383 		'unset X; set -u; echo ${X-GOOD}; echo OK'
    384 	atf_check -s exit:0 -o match:OK -e empty ${TEST_SH} -ce \
    385 		'unset X; set -u; echo ${X-OK}'
    386 	atf_check -s exit:0 -o not-match:ERR -o match:OK -e empty \
    387 		${TEST_SH} -ce 'unset X; set -u; echo ${X+ERR}; echo OK'
    388 
    389 	# and some more ways that are not OK
    390 	atf_check -s not-exit:0 -o not-match:ERR -e not-empty ${TEST_SH} -c \
    391 		'unset X; set -u; echo ${X#foo}; echo ERR'
    392 	atf_check -s not-exit:0 -o not-match:ERR -e not-empty ${TEST_SH} -c \
    393 		'unset X; set -u; echo ${X%%bar}; echo ERR'
    394 
    395 	# lastly, just while we are checking unset vars, test aborts w/o -u
    396 	atf_check -s not-exit:0 -o not-match:ERR -e not-empty ${TEST_SH} -c \
    397 		'unset X; echo ${X?}; echo ERR'
    398 	atf_check -s not-exit:0 -o not-match:ERR -e match:X_NOT_SET \
    399 		${TEST_SH} -c 'unset X; echo ${X?X_NOT_SET}; echo ERR'
    400 }
    401 
    402 atf_test_case set_v
    403 set_v_head() {
    404 	atf_set "descr" "Tests that 'set -v' turns on input read echoing " \
    405 	                "and that it behaves as defined by the standard"
    406 }
    407 set_v_body() {
    408 	test_option_on_off v
    409 
    410 	# check that -v does nothing if no later input line is read
    411 	atf_check -s exit:0 \
    412 			-o match:OKOK -o not-match:echo -o not-match:printf \
    413 			-e empty \
    414 		${TEST_SH} -ec 'printf "%s" OK; set -v; echo OK; exit 0'
    415 
    416 	# but that it does when there are multiple lines
    417 	atf_check -s exit:0 \
    418 			-o match:OKOK -o not-match:echo -o not-match:printf \
    419 			-e match:printf -e match:OK -e match:echo \
    420 		${TEST_SH} -ec '{
    421 					echo "set -v"
    422 					echo "printf %s OK"
    423 					echo "echo OK"
    424 					echo "exit 0"
    425 				} | '"'${TEST_SH}'"' -e'
    426 
    427 	# and that it can be disabled again
    428 	atf_check -s exit:0 \
    429 			-o match:OKOK -o not-match:echo -o not-match:printf \
    430 			-e match:printf -e match:OK -e not-match:echo \
    431 		${TEST_SH} -ec '{
    432 					echo "set -v"
    433 					echo "printf %s OK"
    434 					echo "set +v"
    435 					echo "echo OK"
    436 					echo "exit 0"
    437 				} | '"'${TEST_SH}'"' -e'
    438 
    439 	# and lastly, that shell keywords do get output when "read"
    440 	atf_check -s exit:0 \
    441 			-o match:111222333 -o not-match:printf \
    442 			-o not-match:for -o not-match:do -o not-match:done \
    443 			-e match:printf -e match:111 -e not-match:111222 \
    444 			-e match:for -e match:do -e match:done \
    445 		${TEST_SH} -ec '{
    446 					echo "set -v"
    447 					echo "for i in 111 222 333"
    448 					echo "do"
    449 					echo "printf %s \$i"
    450 					echo "done"
    451 					echo "exit 0"
    452 				} | '"'${TEST_SH}'"' -e'
    453 }
    454 
    455 atf_test_case set_x
    456 set_x_head() {
    457 	atf_set "descr" "Tests that 'set -x' turns on command exec logging " \
    458 	                "and that it behaves as defined by the standard"
    459 }
    460 set_x_body() {
    461 	test_option_on_off x
    462 
    463 	# check that cmd output appears after -x is enabled
    464 	atf_check -s exit:0 \
    465 			-o match:OKOK -o not-match:echo -o not-match:printf \
    466 			-e not-match:printf -e match:OK -e match:echo \
    467 		${TEST_SH} -ec 'printf "%s" OK; set -x; echo OK; exit 0'
    468 
    469 	# and that it stops again afer -x is disabled
    470 	atf_check -s exit:0 \
    471 			-o match:OKOK -o not-match:echo -o not-match:printf \
    472 			-e match:printf -e match:OK -e not-match:echo \
    473 	    ${TEST_SH} -ec 'set -x; printf "%s" OK; set +x; echo OK; exit 0'
    474 
    475 	# also check that PS4 is output correctly
    476 	atf_check -s exit:0 \
    477 			-o match:OK -o not-match:echo \
    478 			-e match:OK -e match:Run:echo \
    479 		${TEST_SH} -ec 'PS4=Run:; set -x; echo OK; exit 0'
    480 
    481 	return 0
    482 
    483 	# This one seems controversial... I suspect it is NetBSD's sh
    484 	# that is wrong to not output "for" "while" "if" ... etc
    485 
    486 	# and lastly, that shell keywords do not get output when "executed"
    487 	atf_check -s exit:0 \
    488 			-o match:111222333 -o not-match:printf \
    489 			-o not-match:for \
    490 			-e match:printf -e match:111 -e not-match:111222 \
    491 			-e not-match:for -e not-match:do -e not-match:done \
    492 		${TEST_SH} -ec \
    493 	   'set -x; for i in 111 222 333; do printf "%s" $i; done; echo; exit 0'
    494 }
    495 
    496 opt_test_setup()
    497 {
    498 	test -n "$1" || { echo >&2 "Internal error"; exit 1; }
    499 
    500 	cat > "$1" << 'END_OF_FUNCTIONS'
    501 local_opt_check()
    502 {
    503 	local -
    504 }
    505 
    506 instr()
    507 {
    508 	expr "$2" : "\(.*$1\)" >/dev/null
    509 }
    510 
    511 save_opts()
    512 {
    513 	local -
    514 
    515 	set -e
    516 	set -u
    517 
    518 	instr e "$-" && instr u "$-" && return 0
    519 	echo ERR
    520 }
    521 
    522 fiddle_opts()
    523 {
    524 	set -e
    525 	set -u
    526 
    527 	instr e "$-" && instr u "$-" && return 0
    528 	echo ERR
    529 }
    530 
    531 local_test()
    532 {
    533 	set +eu
    534 
    535 	save_opts
    536 	instr '[eu]' "$-" || printf %s "OK"
    537 
    538 	fiddle_opts
    539 	instr e "$-" && instr u "$-" && printf %s "OK"
    540 
    541 	set +eu
    542 }
    543 END_OF_FUNCTIONS
    544 }
    545 
    546 atf_test_case restore_local_opts
    547 restore_local_opts_head() {
    548 	atf_set "descr" "Tests that 'local -' saves and restores options.  " \
    549 			"Note that "local" is a local shell addition"
    550 }
    551 restore_local_opts_body() {
    552 	atf_require_prog cat
    553 	atf_require_prog expr
    554 
    555 	FN="test-funcs.$$"
    556 	opt_test_setup "${FN}" || atf_skip "Cannot setup test environment"
    557 
    558 	${TEST_SH} -ec ". './${FN}'; local_opt_check" 2>/dev/null ||
    559 		atf_skip "sh extension 'local -' not supported by ${TEST_SH}"
    560 
    561 	atf_check -s exit:0 -o match:OKOK -o not-match:ERR -e empty \
    562 		${TEST_SH} -ec ". './${FN}'; local_test"
    563 }
    564 
    565 atf_test_case vi_emacs_VE_toggle
    566 vi_emacs_VE_toggle_head() {
    567 	atf_set "descr" "Tests enabling vi disables emacs (and v.v - but why?)"\
    568 			"  Note that -V and -E are local shell additions"
    569 }
    570 vi_emacs_VE_toggle_body() {
    571 
    572 	test_optional_on_off V E ||
    573 	  atf_skip "One or both V & E opts unsupported by ${TEST_SH}"
    574 
    575 	atf_check -s exit:0 -o empty -e empty ${TEST_SH} -c '
    576 		q() {
    577 			eval "case \"$-\" in
    578 			(*${2}*)	return 1;;
    579 			(*${1}*)	return 0;;
    580 			esac"
    581 			return 1
    582 		}
    583 		x() {
    584 			echo >&2 "Option set or toggle failure:" \
    585 					" on=$1 off=$2 set=$-"
    586 			exit 1
    587 		}
    588 		set -V; q V E || x V E
    589 		set -E; q E V || x E V
    590 		set -V; q V E || x V E
    591 		set +EV; q "" "[VE]" || x "" VE
    592 		exit 0
    593 	'
    594 }
    595 
    596 atf_test_case xx_bogus
    597 xx_bogus_head() {
    598 	atf_set "descr" "Tests that attempting to set a nonsense option fails."
    599 }
    600 xx_bogus_body() {
    601 	# Biggest problem here is picking a "nonsense option" that is
    602 	# not implemented by any shell, anywhere.  Hopefully this will do.
    603 	atf_check -s not-exit:0 -o empty -e not-empty \
    604 		${TEST_SH} -c 'set -% ; echo ERR'
    605 }
    606 
    607 atf_test_case Option_switching
    608 Option_switching_head() {
    609 	atf_set "descr" "options can be enabled and disabled"
    610 }
    611 Option_switching_body() {
    612 
    613 	# Cannot test -m, setting it causes test shell to fail...
    614 	# (test shell gets SIGKILL!)  Wonder why ... something related to atf
    615 	# That is, it works if just run as "sh -c 'echo $-; set -m; echo $-'"
    616 
    617 	# Don't bother testing toggling -n, once on, it stays on...
    618 	# (and because the test fn refuses to allow us to try)
    619 
    620 	# Cannot test -o or -c here, or the extension -s
    621 	# they can only be used, not switched
    622 
    623 	# these are the posix options, that all shells should implement
    624 	test_option_on_off a b C e f h u v x      # m
    625 
    626 	# and these are extensions that might not exist (non-fatal to test)
    627 	# -i and -s (and -c) are posix options, but are not required to
    628 	# be accessable via the "set" command, just the command line.
    629 	# We allow for -i to work with set, as that makes some sense,
    630 	# -c and -s do not.
    631 	test_optional_on_off E i I p q V || true
    632 
    633 	# Also test (some) option combinations ...
    634 	# only testing posix options here, because it is easier...
    635 	test_option_on_off aeu vx Ca aCefux
    636 }
    637 
    638 atf_init_test_cases() {
    639 	# tests are run in order sort of names produces, so choose names wisely
    640 
    641 	# this one tests turning on/off all the mandatory. and extra flags
    642 	atf_add_test_case Option_switching
    643 	# and this tests the NetBSD "local -" functionality in functions.
    644 	atf_add_test_case restore_local_opts
    645 
    646 	# no tests for	-m (no idea how to do that one)
    647 	#		-I (no easy way to generate the EOF it ignores)
    648 	#		-i (not sure how to test that one at the minute)
    649 	#		-p (because we aren't going to run tests setuid)
    650 	#		-V/-E (too much effort, and a real test would be huge)
    651 	#		-c (because almost all the other tests test it anyway)
    652 	#		-q (because, for now, I am lazy)
    653 	#		-s (coming soon, hopefully)
    654 	#		-o (really +o: again, hopefully soon)
    655 	#		-o longname (again, just laziness, don't wait...)
    656 	# 		-h/-b (because NetBSD doesn't implement them)
    657 	atf_add_test_case set_a
    658 	atf_add_test_case set_C
    659 	atf_add_test_case set_e
    660 	atf_add_test_case set_f
    661 	atf_add_test_case set_n
    662 	atf_add_test_case set_u
    663 	atf_add_test_case set_v
    664 	atf_add_test_case set_x
    665 
    666 	atf_add_test_case vi_emacs_VE_toggle
    667 	atf_add_test_case xx_bogus
    668 }
    669