t_fsplit.sh revision 1.5 1 1.5 kre # $NetBSD: t_fsplit.sh,v 1.5 2017/06/03 10:27:05 kre Exp $
2 1.1 jruoho #
3 1.2 christos # Copyright (c) 2007-2016 The NetBSD Foundation, Inc.
4 1.1 jruoho # All rights reserved.
5 1.1 jruoho #
6 1.1 jruoho # Redistribution and use in source and binary forms, with or without
7 1.1 jruoho # modification, are permitted provided that the following conditions
8 1.1 jruoho # are met:
9 1.1 jruoho # 1. Redistributions of source code must retain the above copyright
10 1.1 jruoho # notice, this list of conditions and the following disclaimer.
11 1.1 jruoho # 2. Redistributions in binary form must reproduce the above copyright
12 1.1 jruoho # notice, this list of conditions and the following disclaimer in the
13 1.1 jruoho # documentation and/or other materials provided with the distribution.
14 1.1 jruoho #
15 1.1 jruoho # THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
16 1.1 jruoho # ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
17 1.1 jruoho # TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18 1.1 jruoho # PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
19 1.1 jruoho # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
20 1.1 jruoho # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
21 1.1 jruoho # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22 1.1 jruoho # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23 1.1 jruoho # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
24 1.1 jruoho # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25 1.1 jruoho # POSSIBILITY OF SUCH DAMAGE.
26 1.1 jruoho #
27 1.1 jruoho
28 1.1 jruoho # The standard
29 1.1 jruoho # http://www.opengroup.org/onlinepubs/009695399/utilities/xcu_chap02.html
30 1.1 jruoho # explains (section 2.6) that Field splitting should be performed on the
31 1.1 jruoho # result of variable expansions.
32 1.1 jruoho # In particular this means that in ${x-word}, 'word' must be expanded as if
33 1.1 jruoho # the "${x-" and "}" were absent from the input line.
34 1.1 jruoho #
35 1.1 jruoho # So: sh -c 'set ${x-a b c}; echo $#' should give 3.
36 1.2 christos # and: sh -c 'set -- ${x-}' echo $#' shold give 0
37 1.1 jruoho #
38 1.1 jruoho
39 1.3 christos # the implementation of "sh" to test
40 1.3 christos : ${TEST_SH:="/bin/sh"}
41 1.3 christos
42 1.1 jruoho nl='
43 1.1 jruoho '
44 1.1 jruoho
45 1.1 jruoho check()
46 1.1 jruoho {
47 1.2 christos TEST=$((${TEST} + 1))
48 1.2 christos
49 1.2 christos case "$#" in
50 1.2 christos (2) ;;
51 1.2 christos (*) atf_fail "Internal test error, $# args to check test ${TEST}";;
52 1.2 christos esac
53 1.2 christos
54 1.2 christos result=$( ${TEST_SH} -c "unset x; $1" )
55 1.2 christos STATUS="$?"
56 1.2 christos
57 1.1 jruoho # Remove newlines
58 1.1 jruoho oifs="$IFS"
59 1.1 jruoho IFS="$nl"
60 1.1 jruoho result="$(echo $result)"
61 1.1 jruoho IFS="$oifs"
62 1.2 christos
63 1.2 christos # trim the test text in case we use it in a message below
64 1.2 christos case "$1" in
65 1.2 christos ????????????????*)
66 1.2 christos set -- "$(expr "$1" : '\(............\).*')..." "$2" ;;
67 1.2 christos esac
68 1.2 christos
69 1.1 jruoho if [ "$2" != "$result" ]
70 1.1 jruoho then
71 1.2 christos if [ "${STATUS}" = "0" ]
72 1.2 christos then
73 1.2 christos atf_fail "Test ${TEST} '$1': expected [$2], found [$result]"
74 1.2 christos else
75 1.2 christos atf_fail \
76 1.2 christos "TEST ${TEST} '$1' failed ($STATUS): expected [$2], found [$result]"
77 1.2 christos fi
78 1.2 christos elif [ "${STATUS}" != 0 ]
79 1.2 christos then
80 1.2 christos atf_fail "TEST ${TEST} '$1' failed ($STATUS)"
81 1.1 jruoho fi
82 1.1 jruoho }
83 1.1 jruoho
84 1.1 jruoho atf_test_case for
85 1.1 jruoho for_head() {
86 1.1 jruoho atf_set "descr" "Checks field splitting in for loops"
87 1.1 jruoho }
88 1.1 jruoho for_body() {
89 1.1 jruoho unset x
90 1.1 jruoho
91 1.2 christos TEST=0
92 1.1 jruoho # Since I managed to break this, leave the test in
93 1.1 jruoho check 'for f in $x; do echo x${f}y; done' ''
94 1.1 jruoho }
95 1.1 jruoho
96 1.1 jruoho atf_test_case default_val
97 1.1 jruoho default_val_head() {
98 1.1 jruoho atf_set "descr" "Checks field splitting in variable default values"
99 1.1 jruoho }
100 1.1 jruoho default_val_body() {
101 1.2 christos TEST=0
102 1.1 jruoho # Check that IFS is applied to text from ${x-...} unless it is inside
103 1.1 jruoho # any set of "..."
104 1.2 christos check 'set -- ${x-a b c}; echo $#' 3
105 1.2 christos
106 1.2 christos check 'set -- ${x-"a b" c}; echo $#' 2
107 1.2 christos check 'set -- ${x-a "b c"}; echo $#' 2
108 1.2 christos check 'set -- ${x-"a b c"}; echo $#' 1
109 1.2 christos
110 1.2 christos check "set -- \${x-'a b' c}; echo \$#" 2
111 1.2 christos check "set -- \${x-a 'b c'}; echo \$#" 2
112 1.2 christos check "set -- \${x-'a b c'}; echo \$#" 1
113 1.2 christos
114 1.2 christos check 'set -- ${x-a\ b c}; echo $#' 2
115 1.2 christos check 'set -- ${x-a b\ c}; echo $#' 2
116 1.2 christos check 'set -- ${x-a\ b\ c}; echo $#' 1
117 1.2 christos
118 1.2 christos check 'set -- ${x}; echo $#' 0
119 1.2 christos check 'set -- ${x-}; echo $#' 0
120 1.2 christos check 'set -- ${x-""}; echo $#' 1
121 1.2 christos check 'set -- ""${x}; echo $#' 1
122 1.2 christos check 'set -- ""${x-}; echo $#' 1
123 1.2 christos check 'set -- ""${x-""}; echo $#' 1
124 1.2 christos check 'set -- ${x}""; echo $#' 1
125 1.2 christos check 'set -- ${x-}""; echo $#' 1
126 1.2 christos check 'set -- ${x-""}""; echo $#' 1
127 1.2 christos check 'set -- ""${x}""; echo $#' 1
128 1.2 christos check 'set -- ""${x-}""; echo $#' 1
129 1.2 christos check 'set -- ""${x-""}""; echo $#' 1
130 1.2 christos
131 1.2 christos check 'for i in ${x-a b c}; do echo "z${i}z"; done' \
132 1.2 christos 'zaz zbz zcz'
133 1.2 christos check 'for i in ${x-"a b" c}; do echo "z${i}z"; done' \
134 1.2 christos 'za bz zcz'
135 1.2 christos check 'for i in ${x-"a ${x-b c}" d}; do echo "z${i}z"; done' \
136 1.2 christos 'za b cz zdz'
137 1.2 christos check 'for i in ${x-a ${x-b c} d}; do echo "z${i}z"; done' \
138 1.2 christos 'zaz zbz zcz zdz'
139 1.2 christos
140 1.5 kre # I am not sure the first of these two is correct, the rules on
141 1.5 kre # quoting word in ${var-word} are peculiar, and hard to fathom...
142 1.5 kre # It is what the NetBSD shell does, and bash, not the freebsd shell
143 1.5 kre # and not ksh93 (as of Mar 1, 2016, and still in June 2017)
144 1.5 kre # The likely correct interp of the next one is 'za bz zcz zdz'
145 1.2 christos
146 1.5 kre # should be: uuuu qqqqqq uuu q uuu (unquoted/quoted) no nesting.
147 1.2 christos check 'for i in ${x-"a ${x-"b c"}" d}; do echo "z${i}z"; done' \
148 1.2 christos 'za b cz zdz'
149 1.2 christos check 'for i in ${x-a ${x-"b c"} d}; do echo "z${i}z"; done' \
150 1.2 christos 'zaz zb cz zdz'
151 1.2 christos }
152 1.2 christos
153 1.2 christos atf_test_case replacement_val
154 1.2 christos replacement_val_head() {
155 1.2 christos atf_set "descr" "Checks field splitting in variable replacement values"
156 1.2 christos }
157 1.2 christos replacement_val_body() {
158 1.2 christos TEST=0
159 1.2 christos
160 1.2 christos # Check that IFS is applied to text from ${x+...} unless it is inside
161 1.2 christos # any set of "...", or whole expansion is quoted, or both...
162 1.2 christos
163 1.2 christos check 'x=BOGUS; set -- ${x+a b c}; echo $#' 3
164 1.2 christos
165 1.2 christos check 'x=BOGUS; set -- ${x+"a b" c}; echo $#' 2
166 1.2 christos check 'x=BOGUS; set -- ${x+a "b c"}; echo $#' 2
167 1.2 christos check 'x=BOGUS; set -- ${x+"a b c"}; echo $#' 1
168 1.2 christos
169 1.2 christos check "x=BOGUS; set -- \${x+'a b' c}; echo \$#" 2
170 1.2 christos check "x=BOGUS; set -- \${x+a 'b c'}; echo \$#" 2
171 1.2 christos check "x=BOGUS; set -- \${x+'a b c'}; echo \$#" 1
172 1.2 christos
173 1.2 christos check 'x=BOGUS; set -- ${x+a\ b c}; echo $#' 2
174 1.2 christos check 'x=BOGUS; set -- ${x+a b\ c}; echo $#' 2
175 1.2 christos check 'x=BOGUS; set -- ${x+a\ b\ c}; echo $#' 1
176 1.2 christos
177 1.2 christos check 'x=BOGUS; set -- ${x+}; echo $#' 0
178 1.2 christos check 'x=BOGUS; set -- ${x+""}; echo $#' 1
179 1.2 christos check 'x=BOGUS; set -- ""${x+}; echo $#' 1
180 1.2 christos check 'x=BOGUS; set -- ""${x+""}; echo $#' 1
181 1.2 christos check 'x=BOGUS; set -- ${x+}""; echo $#' 1
182 1.2 christos check 'x=BOGUS; set -- ${x+""}""; echo $#' 1
183 1.2 christos check 'x=BOGUS; set -- ""${x+}""; echo $#' 1
184 1.2 christos check 'x=BOGUS; set -- ""${x+""}""; echo $#' 1
185 1.2 christos
186 1.2 christos # verify that the value of $x does not affecty the value of ${x+...}
187 1.2 christos check 'x=BOGUS; set -- ${x+}; echo X$1' X
188 1.2 christos check 'x=BOGUS; set -- ${x+""}; echo X$1' X
189 1.2 christos check 'x=BOGUS; set -- ""${x+}; echo X$1' X
190 1.2 christos check 'x=BOGUS; set -- ""${x+""}; echo X$1' X
191 1.2 christos check 'x=BOGUS; set -- ${x+}""; echo X$1' X
192 1.2 christos check 'x=BOGUS; set -- ${x+""}""; echo X$1' X
193 1.2 christos check 'x=BOGUS; set -- ""${x+}""; echo X$1' X
194 1.2 christos check 'x=BOGUS; set -- ""${x+""}""; echo X$1' X
195 1.2 christos
196 1.2 christos check 'x=BOGUS; set -- ${x+}; echo X${1-:}X' X:X
197 1.2 christos check 'x=BOGUS; set -- ${x+""}; echo X${1-:}X' XX
198 1.2 christos check 'x=BOGUS; set -- ""${x+}; echo X${1-:}X' XX
199 1.2 christos check 'x=BOGUS; set -- ""${x+""}; echo X${1-:}X' XX
200 1.2 christos check 'x=BOGUS; set -- ${x+}""; echo X${1-:}X' XX
201 1.2 christos check 'x=BOGUS; set -- ${x+""}""; echo X${1-:}X' XX
202 1.2 christos check 'x=BOGUS; set -- ""${x+}""; echo X${1-:}X' XX
203 1.2 christos check 'x=BOGUS; set -- ""${x+""}""; echo X${1-:}X' XX
204 1.2 christos
205 1.2 christos # and validate that the replacement can be used as expected
206 1.2 christos check 'x=BOGUS; for i in ${x+a b c}; do echo "z${i}z"; done'\
207 1.2 christos 'zaz zbz zcz'
208 1.2 christos check 'x=BOGUS; for i in ${x+"a b" c}; do echo "z${i}z"; done'\
209 1.2 christos 'za bz zcz'
210 1.2 christos check 'x=BOGUS; for i in ${x+"a ${x+b c}" d}; do echo "z${i}z"; done'\
211 1.2 christos 'za b cz zdz'
212 1.2 christos check 'x=BOGUS; for i in ${x+"a ${x+"b c"}" d}; do echo "z${i}z"; done'\
213 1.2 christos 'za b cz zdz'
214 1.2 christos check 'x=BOGUS; for i in ${x+a ${x+"b c"} d}; do echo "z${i}z"; done'\
215 1.2 christos 'zaz zb cz zdz'
216 1.2 christos check 'x=BOGUS; for i in ${x+a ${x+b c} d}; do echo "z${i}z"; done'\
217 1.2 christos 'zaz zbz zcz zdz'
218 1.1 jruoho }
219 1.1 jruoho
220 1.1 jruoho atf_test_case ifs_alpha
221 1.1 jruoho ifs_alpha_head() {
222 1.1 jruoho atf_set "descr" "Checks that field splitting works with alphabetic" \
223 1.1 jruoho "characters"
224 1.1 jruoho }
225 1.1 jruoho ifs_alpha_body() {
226 1.1 jruoho unset x
227 1.1 jruoho
228 1.2 christos TEST=0
229 1.1 jruoho # repeat with an alphabetic in IFS
230 1.1 jruoho check 'IFS=q; set ${x-aqbqc}; echo $#' 3
231 1.2 christos check 'IFS=q; for i in ${x-aqbqc}; do echo "z${i}z"; done' \
232 1.2 christos 'zaz zbz zcz'
233 1.2 christos check 'IFS=q; for i in ${x-"aqb"qc}; do echo "z${i}z"; done' \
234 1.2 christos 'zaqbz zcz'
235 1.2 christos check 'IFS=q; for i in ${x-"aq${x-bqc}"qd}; do echo "z${i}z"; done' \
236 1.2 christos 'zaqbqcz zdz'
237 1.2 christos check 'IFS=q; for i in ${x-"aq${x-"bqc"}"qd}; do echo "z${i}z"; done' \
238 1.2 christos 'zaqbqcz zdz'
239 1.2 christos check 'IFS=q; for i in ${x-aq${x-"bqc"}qd}; do echo "z${i}z"; done' \
240 1.2 christos 'zaz zbqcz zdz'
241 1.1 jruoho }
242 1.1 jruoho
243 1.1 jruoho atf_test_case quote
244 1.1 jruoho quote_head() {
245 1.1 jruoho atf_set "descr" "Checks that field splitting works with multi-word" \
246 1.1 jruoho "fields"
247 1.1 jruoho }
248 1.1 jruoho quote_body() {
249 1.1 jruoho unset x
250 1.1 jruoho
251 1.2 christos TEST=0
252 1.1 jruoho # Some quote propagation checks
253 1.1 jruoho check 'set "${x-a b c}"; echo $#' 1
254 1.1 jruoho check 'set "${x-"a b" c}"; echo $1' 'a b c'
255 1.1 jruoho check 'for i in "${x-a b c}"; do echo "z${i}z"; done' 'za b cz'
256 1.1 jruoho }
257 1.1 jruoho
258 1.1 jruoho atf_test_case dollar_at
259 1.1 jruoho dollar_at_head() {
260 1.1 jruoho atf_set "descr" "Checks that field splitting works when expanding" \
261 1.1 jruoho "\$@"
262 1.1 jruoho }
263 1.1 jruoho dollar_at_body() {
264 1.1 jruoho unset x
265 1.1 jruoho
266 1.2 christos TEST=0
267 1.1 jruoho # Check we get "$@" right
268 1.2 christos
269 1.2 christos check 'set --; for i in x"$@"x; do echo "z${i}z"; done' 'zxxz'
270 1.2 christos check 'set a; for i in x"$@"x; do echo "z${i}z"; done' 'zxaxz'
271 1.2 christos check 'set a b; for i in x"$@"x; do echo "z${i}z"; done' 'zxaz zbxz'
272 1.2 christos
273 1.2 christos check 'set --; for i; do echo "z${i}z"; done' ''
274 1.2 christos check 'set --; for i in $@; do echo "z${i}z"; done' ''
275 1.2 christos check 'set --; for i in "$@"; do echo "z${i}z"; done' ''
276 1.2 christos # atf_expect_fail "PR bin/50834"
277 1.2 christos check 'set --; for i in ""$@; do echo "z${i}z"; done' 'zz'
278 1.2 christos # atf_expect_pass
279 1.2 christos check 'set --; for i in $@""; do echo "z${i}z"; done' 'zz'
280 1.2 christos check 'set --; for i in ""$@""; do echo "z${i}z"; done' 'zz'
281 1.2 christos check 'set --; for i in """$@"; do echo "z${i}z"; done' 'zz'
282 1.2 christos check 'set --; for i in "$@"""; do echo "z${i}z"; done' 'zz'
283 1.2 christos check 'set --; for i in """$@""";do echo "z${i}z"; done' 'zz'
284 1.2 christos
285 1.2 christos check 'set ""; for i; do echo "z${i}z"; done' 'zz'
286 1.2 christos check 'set ""; for i in "$@"; do echo "z${i}z"; done' 'zz'
287 1.2 christos check 'set "" ""; for i; do echo "z${i}z"; done' 'zz zz'
288 1.2 christos check 'set "" ""; for i in "$@"; do echo "z${i}z"; done' 'zz zz'
289 1.2 christos check 'set "" ""; for i in $@; do echo "z${i}z"; done' ''
290 1.2 christos
291 1.2 christos check 'set "a b" c; for i; do echo "z${i}z"; done' \
292 1.2 christos 'za bz zcz'
293 1.2 christos check 'set "a b" c; for i in "$@"; do echo "z${i}z"; done' \
294 1.2 christos 'za bz zcz'
295 1.2 christos check 'set "a b" c; for i in $@; do echo "z${i}z"; done' \
296 1.2 christos 'zaz zbz zcz'
297 1.2 christos check 'set " a b " c; for i in "$@"; do echo "z${i}z"; done' \
298 1.2 christos 'z a b z zcz'
299 1.4 christos
300 1.4 christos check 'set a b c; for i in "$@$@"; do echo "z${i}z"; done' \
301 1.4 christos 'zaz zbz zcaz zbz zcz'
302 1.4 christos check 'set a b c; for i in "$@""$@";do echo "z${i}z"; done' \
303 1.4 christos 'zaz zbz zcaz zbz zcz'
304 1.1 jruoho }
305 1.1 jruoho
306 1.1 jruoho atf_test_case ifs
307 1.1 jruoho ifs_head() {
308 1.1 jruoho atf_set "descr" "Checks that IFS correctly configures field" \
309 1.1 jruoho "splitting behavior"
310 1.1 jruoho }
311 1.1 jruoho ifs_body() {
312 1.1 jruoho unset x
313 1.1 jruoho
314 1.2 christos TEST=0
315 1.1 jruoho # Some IFS tests
316 1.1 jruoho check 't="-- "; IFS=" "; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '0'
317 1.1 jruoho check 't=" x"; IFS=" x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '1'
318 1.1 jruoho check 't=" x "; IFS=" x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '1'
319 1.1 jruoho check 't=axb; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '2 a:b'
320 1.1 jruoho check 't="a x b"; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '2 a : b'
321 1.1 jruoho check 't="a xx b"; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '3 a :: b'
322 1.1 jruoho check 't="a xx b"; IFS="x "; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '3 a::b'
323 1.1 jruoho # A recent 'clarification' means that a single trailing IFS non-whitespace
324 1.1 jruoho # doesn't generate an empty parameter
325 1.1 jruoho check 't="xax"; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '2 :a'
326 1.1 jruoho check 't="xax "; IFS="x "; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '2 :a'
327 1.1 jruoho # Verify that IFS isn't being applied where it shouldn't be.
328 1.1 jruoho check 'IFS="x"; set axb; IFS=":"; r="$*"; IFS=; echo $# $r' '1 axb'
329 1.1 jruoho }
330 1.1 jruoho
331 1.1 jruoho atf_test_case var_length
332 1.1 jruoho var_length_head() {
333 1.1 jruoho atf_set "descr" "Checks that field splitting works when expanding" \
334 1.1 jruoho "a variable's length"
335 1.1 jruoho }
336 1.1 jruoho var_length_body() {
337 1.2 christos TEST=0
338 1.1 jruoho
339 1.1 jruoho long=12345678123456781234567812345678
340 1.1 jruoho long=$long$long$long$long
341 1.2 christos export long
342 1.5 kre unset x
343 1.2 christos
344 1.2 christos # first test that the test method works...
345 1.2 christos check 'set -u; : ${long}; echo ${#long}' '128'
346 1.2 christos
347 1.2 christos # Check that we apply IFS to ${#var}
348 1.2 christos check 'echo ${#long}; IFS=2; echo ${#long}; set 1 ${#long};echo $#' \
349 1.2 christos '128 1 8 3'
350 1.5 kre check 'IFS=2; set ${x-${#long}}; IFS=" "; echo $* $#' '1 8 2'
351 1.5 kre check 'IFS=2; set ${x-"${#long}"}; IFS=" "; echo $* $#' '128 1'
352 1.5 kre check 'IFS=2; set "${x-${#long}}"; IFS=" "; echo $* $#' '128 1'
353 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo $* $#' '1 8 '
354 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo $* "$#"' '1 8 2'
355 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo "$*" "$#"' '128 2'
356 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo "$@" "$#"' '1 8 2'
357 1.1 jruoho }
358 1.1 jruoho
359 1.1 jruoho atf_init_test_cases() {
360 1.1 jruoho atf_add_test_case for
361 1.1 jruoho atf_add_test_case default_val
362 1.2 christos atf_add_test_case replacement_val
363 1.1 jruoho atf_add_test_case ifs_alpha
364 1.1 jruoho atf_add_test_case quote
365 1.1 jruoho atf_add_test_case dollar_at
366 1.1 jruoho atf_add_test_case ifs
367 1.1 jruoho atf_add_test_case var_length
368 1.1 jruoho }
369