t_fsplit.sh revision 1.10 1 1.10 kre # $NetBSD: t_fsplit.sh,v 1.10 2024/10/19 11:59:51 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.6 kre # and: sh -c 'set -- ${x-}' echo $#' should 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.9 kre if [ "${TEST}" -eq 0 ]
48 1.9 kre then
49 1.9 kre FAILURES=
50 1.9 kre fi
51 1.9 kre
52 1.2 christos TEST=$((${TEST} + 1))
53 1.2 christos
54 1.2 christos case "$#" in
55 1.2 christos (2) ;;
56 1.9 kre (*) atf_fail "Internal test error, $# args to check, test ${TEST}";;
57 1.2 christos esac
58 1.2 christos
59 1.8 kre result=$( ${TEST_SH} -c "unset x a b d c e f g h; $1" )
60 1.2 christos STATUS="$?"
61 1.2 christos
62 1.1 jruoho # Remove newlines
63 1.1 jruoho oifs="$IFS"
64 1.1 jruoho IFS="$nl"
65 1.1 jruoho result="$(echo $result)"
66 1.1 jruoho IFS="$oifs"
67 1.2 christos
68 1.9 kre # # trim the test text in case we use it in a message below
69 1.9 kre # case "$1" in
70 1.9 kre # ????????????????*)
71 1.9 kre # set -- "$(expr "$1" : '\(............\).*')..." "$2" ;;
72 1.9 kre # esac
73 1.2 christos
74 1.9 kre if [ "$2" != "${result}" ]
75 1.1 jruoho then
76 1.9 kre FAILURES="${FAILURES}${FAILURES:+ }${TEST}"
77 1.9 kre printf >&2 'Sub-test %d failed:\n %s\n' \
78 1.9 kre "${TEST}" "$1"
79 1.9 kre printf >&2 ' Expected: [%s]\n' "$2"
80 1.9 kre printf >&2 ' Received: [%s]\n' "${result}"
81 1.9 kre
82 1.9 kre if [ "${STATUS}" != 0 ]
83 1.2 christos then
84 1.9 kre printf >&2 ' Sub-test exit status: %d\n' "${STATUS}"
85 1.2 christos fi
86 1.2 christos elif [ "${STATUS}" != 0 ]
87 1.2 christos then
88 1.9 kre FAILURES="${FAILURES}${FAILURES:+ }${TEST}"
89 1.9 kre printf >&2 'Sub-test %d failed:\n\t%s\n' \
90 1.9 kre "${TEST}" "$1"
91 1.9 kre printf >&2 ' Sub-test exit status: %d (with correct output)\n' \
92 1.9 kre "${STATUS}"
93 1.9 kre fi
94 1.9 kre
95 1.9 kre return 0
96 1.9 kre }
97 1.9 kre
98 1.9 kre check_results()
99 1.9 kre {
100 1.9 kre NAME=$1
101 1.9 kre
102 1.9 kre set -- ${FAILURES}
103 1.9 kre
104 1.9 kre if [ $# -eq 0 ]
105 1.9 kre then
106 1.9 kre return 0
107 1.1 jruoho fi
108 1.6 kre
109 1.9 kre unset IFS
110 1.9 kre printf >&2 'Subtest %s: %d sub-tests (of %d) [%s] failed.\n' \
111 1.9 kre "${NAME}" "$#" "${TEST}" "$*"
112 1.9 kre
113 1.9 kre atf_fail "$# of ${TEST} sub-tests (${FAILURES}), see stderr"
114 1.6 kre return 0
115 1.1 jruoho }
116 1.1 jruoho
117 1.1 jruoho atf_test_case for
118 1.9 kre for_head()
119 1.9 kre {
120 1.1 jruoho atf_set "descr" "Checks field splitting in for loops"
121 1.1 jruoho }
122 1.9 kre for_body()
123 1.9 kre {
124 1.1 jruoho unset x
125 1.1 jruoho
126 1.2 christos TEST=0
127 1.1 jruoho # Since I managed to break this, leave the test in
128 1.1 jruoho check 'for f in $x; do echo x${f}y; done' ''
129 1.9 kre
130 1.9 kre check_results for
131 1.1 jruoho }
132 1.1 jruoho
133 1.1 jruoho atf_test_case default_val
134 1.9 kre default_val_head()
135 1.9 kre {
136 1.1 jruoho atf_set "descr" "Checks field splitting in variable default values"
137 1.1 jruoho }
138 1.9 kre default_val_body()
139 1.9 kre {
140 1.2 christos TEST=0
141 1.1 jruoho # Check that IFS is applied to text from ${x-...} unless it is inside
142 1.1 jruoho # any set of "..."
143 1.2 christos check 'set -- ${x-a b c}; echo $#' 3
144 1.2 christos
145 1.2 christos check 'set -- ${x-"a b" c}; echo $#' 2
146 1.2 christos check 'set -- ${x-a "b c"}; echo $#' 2
147 1.2 christos check 'set -- ${x-"a b c"}; echo $#' 1
148 1.2 christos
149 1.2 christos check "set -- \${x-'a b' c}; echo \$#" 2
150 1.2 christos check "set -- \${x-a 'b c'}; echo \$#" 2
151 1.2 christos check "set -- \${x-'a b c'}; echo \$#" 1
152 1.2 christos
153 1.2 christos check 'set -- ${x-a\ b c}; echo $#' 2
154 1.2 christos check 'set -- ${x-a b\ c}; echo $#' 2
155 1.2 christos check 'set -- ${x-a\ b\ c}; echo $#' 1
156 1.2 christos
157 1.2 christos check 'set -- ${x}; echo $#' 0
158 1.2 christos check 'set -- ${x-}; echo $#' 0
159 1.2 christos check 'set -- ${x-""}; echo $#' 1
160 1.2 christos check 'set -- ""${x}; echo $#' 1
161 1.2 christos check 'set -- ""${x-}; echo $#' 1
162 1.2 christos check 'set -- ""${x-""}; echo $#' 1
163 1.2 christos check 'set -- ${x}""; echo $#' 1
164 1.2 christos check 'set -- ${x-}""; echo $#' 1
165 1.2 christos check 'set -- ${x-""}""; echo $#' 1
166 1.2 christos check 'set -- ""${x}""; echo $#' 1
167 1.2 christos check 'set -- ""${x-}""; echo $#' 1
168 1.2 christos check 'set -- ""${x-""}""; echo $#' 1
169 1.2 christos
170 1.2 christos check 'for i in ${x-a b c}; do echo "z${i}z"; done' \
171 1.2 christos 'zaz zbz zcz'
172 1.2 christos check 'for i in ${x-"a b" c}; do echo "z${i}z"; done' \
173 1.2 christos 'za bz zcz'
174 1.2 christos check 'for i in ${x-"a ${x-b c}" d}; do echo "z${i}z"; done' \
175 1.2 christos 'za b cz zdz'
176 1.2 christos check 'for i in ${x-a ${x-b c} d}; do echo "z${i}z"; done' \
177 1.2 christos 'zaz zbz zcz zdz'
178 1.2 christos
179 1.5 kre # I am not sure the first of these two is correct, the rules on
180 1.5 kre # quoting word in ${var-word} are peculiar, and hard to fathom...
181 1.5 kre # It is what the NetBSD shell does, and bash, not the freebsd shell
182 1.5 kre # and not ksh93 (as of Mar 1, 2016, and still in June 2017)
183 1.5 kre # The likely correct interp of the next one is 'za bz zcz zdz'
184 1.2 christos
185 1.7 kre # That and the "should be" below are correct as of POSIX 7 TC2
186 1.7 kre # But this is going to change to "unspecified" in POSIX 8
187 1.7 kre # (resolution of bug 221) so instead of being incorrect (as now)
188 1.7 kre # the NetBSD shell will simply be implementing is version
189 1.7 kre # of unspecified behaviour. Just beware that shells differ,
190 1.7 kre # a shell that fails this test is not incorrect because of it.
191 1.7 kre
192 1.5 kre # should be: uuuu qqqqqq uuu q uuu (unquoted/quoted) no nesting.
193 1.2 christos check 'for i in ${x-"a ${x-"b c"}" d}; do echo "z${i}z"; done' \
194 1.2 christos 'za b cz zdz'
195 1.2 christos check 'for i in ${x-a ${x-"b c"} d}; do echo "z${i}z"; done' \
196 1.2 christos 'zaz zb cz zdz'
197 1.9 kre
198 1.9 kre check_results default_val
199 1.2 christos }
200 1.2 christos
201 1.2 christos atf_test_case replacement_val
202 1.9 kre replacement_val_head()
203 1.9 kre {
204 1.2 christos atf_set "descr" "Checks field splitting in variable replacement values"
205 1.2 christos }
206 1.9 kre replacement_val_body()
207 1.9 kre {
208 1.2 christos TEST=0
209 1.2 christos
210 1.2 christos # Check that IFS is applied to text from ${x+...} unless it is inside
211 1.2 christos # any set of "...", or whole expansion is quoted, or both...
212 1.2 christos
213 1.2 christos check 'x=BOGUS; set -- ${x+a b c}; echo $#' 3
214 1.2 christos
215 1.2 christos check 'x=BOGUS; set -- ${x+"a b" c}; echo $#' 2
216 1.2 christos check 'x=BOGUS; set -- ${x+a "b c"}; echo $#' 2
217 1.2 christos check 'x=BOGUS; set -- ${x+"a b c"}; echo $#' 1
218 1.2 christos
219 1.2 christos check "x=BOGUS; set -- \${x+'a b' c}; echo \$#" 2
220 1.2 christos check "x=BOGUS; set -- \${x+a 'b c'}; echo \$#" 2
221 1.2 christos check "x=BOGUS; set -- \${x+'a b c'}; echo \$#" 1
222 1.2 christos
223 1.2 christos check 'x=BOGUS; set -- ${x+a\ b c}; echo $#' 2
224 1.2 christos check 'x=BOGUS; set -- ${x+a b\ c}; echo $#' 2
225 1.2 christos check 'x=BOGUS; set -- ${x+a\ b\ c}; echo $#' 1
226 1.2 christos
227 1.2 christos check 'x=BOGUS; set -- ${x+}; echo $#' 0
228 1.2 christos check 'x=BOGUS; set -- ${x+""}; echo $#' 1
229 1.2 christos check 'x=BOGUS; set -- ""${x+}; echo $#' 1
230 1.2 christos check 'x=BOGUS; set -- ""${x+""}; echo $#' 1
231 1.2 christos check 'x=BOGUS; set -- ${x+}""; echo $#' 1
232 1.2 christos check 'x=BOGUS; set -- ${x+""}""; echo $#' 1
233 1.2 christos check 'x=BOGUS; set -- ""${x+}""; echo $#' 1
234 1.2 christos check 'x=BOGUS; set -- ""${x+""}""; echo $#' 1
235 1.2 christos
236 1.2 christos # verify that the value of $x does not affecty the value of ${x+...}
237 1.2 christos check 'x=BOGUS; set -- ${x+}; echo X$1' X
238 1.2 christos check 'x=BOGUS; set -- ${x+""}; echo X$1' X
239 1.2 christos check 'x=BOGUS; set -- ""${x+}; echo X$1' X
240 1.2 christos check 'x=BOGUS; set -- ""${x+""}; echo X$1' X
241 1.2 christos check 'x=BOGUS; set -- ${x+}""; echo X$1' X
242 1.2 christos check 'x=BOGUS; set -- ${x+""}""; echo X$1' X
243 1.2 christos check 'x=BOGUS; set -- ""${x+}""; echo X$1' X
244 1.2 christos check 'x=BOGUS; set -- ""${x+""}""; echo X$1' X
245 1.2 christos
246 1.2 christos check 'x=BOGUS; set -- ${x+}; echo X${1-:}X' X:X
247 1.2 christos check 'x=BOGUS; set -- ${x+""}; echo X${1-:}X' XX
248 1.2 christos check 'x=BOGUS; set -- ""${x+}; echo X${1-:}X' XX
249 1.2 christos check 'x=BOGUS; set -- ""${x+""}; echo X${1-:}X' XX
250 1.2 christos check 'x=BOGUS; set -- ${x+}""; echo X${1-:}X' XX
251 1.2 christos check 'x=BOGUS; set -- ${x+""}""; echo X${1-:}X' XX
252 1.2 christos check 'x=BOGUS; set -- ""${x+}""; echo X${1-:}X' XX
253 1.2 christos check 'x=BOGUS; set -- ""${x+""}""; echo X${1-:}X' XX
254 1.2 christos
255 1.2 christos # and validate that the replacement can be used as expected
256 1.2 christos check 'x=BOGUS; for i in ${x+a b c}; do echo "z${i}z"; done'\
257 1.2 christos 'zaz zbz zcz'
258 1.2 christos check 'x=BOGUS; for i in ${x+"a b" c}; do echo "z${i}z"; done'\
259 1.2 christos 'za bz zcz'
260 1.2 christos check 'x=BOGUS; for i in ${x+"a ${x+b c}" d}; do echo "z${i}z"; done'\
261 1.2 christos 'za b cz zdz'
262 1.6 kre
263 1.7 kre # see the (extended) comment in the default_val test. This will be
264 1.7 kre # unspecified, hence we are OK (will be) but expect differences.
265 1.6 kre # also incorrect: uuuu qqqqqq uuu q uuu
266 1.2 christos check 'x=BOGUS; for i in ${x+"a ${x+"b c"}" d}; do echo "z${i}z"; done'\
267 1.2 christos 'za b cz zdz'
268 1.6 kre
269 1.2 christos check 'x=BOGUS; for i in ${x+a ${x+"b c"} d}; do echo "z${i}z"; done'\
270 1.2 christos 'zaz zb cz zdz'
271 1.2 christos check 'x=BOGUS; for i in ${x+a ${x+b c} d}; do echo "z${i}z"; done'\
272 1.2 christos 'zaz zbz zcz zdz'
273 1.9 kre
274 1.9 kre check_results replacement_val
275 1.1 jruoho }
276 1.1 jruoho
277 1.1 jruoho atf_test_case ifs_alpha
278 1.9 kre ifs_alpha_head()
279 1.9 kre {
280 1.1 jruoho atf_set "descr" "Checks that field splitting works with alphabetic" \
281 1.1 jruoho "characters"
282 1.1 jruoho }
283 1.9 kre ifs_alpha_body()
284 1.9 kre {
285 1.1 jruoho unset x
286 1.1 jruoho
287 1.2 christos TEST=0
288 1.1 jruoho # repeat with an alphabetic in IFS
289 1.1 jruoho check 'IFS=q; set ${x-aqbqc}; echo $#' 3
290 1.2 christos check 'IFS=q; for i in ${x-aqbqc}; do echo "z${i}z"; done' \
291 1.2 christos 'zaz zbz zcz'
292 1.2 christos check 'IFS=q; for i in ${x-"aqb"qc}; do echo "z${i}z"; done' \
293 1.2 christos 'zaqbz zcz'
294 1.2 christos check 'IFS=q; for i in ${x-"aq${x-bqc}"qd}; do echo "z${i}z"; done' \
295 1.2 christos 'zaqbqcz zdz'
296 1.6 kre
297 1.6 kre # this is another almost certainly incorrect expectation
298 1.7 kre # (but again, see comment in default_val test - becoming unspecified.)
299 1.6 kre # uu qqqqqq uuu q uu (quoted/unquoted)
300 1.2 christos check 'IFS=q; for i in ${x-"aq${x-"bqc"}"qd}; do echo "z${i}z"; done' \
301 1.2 christos 'zaqbqcz zdz'
302 1.6 kre
303 1.2 christos check 'IFS=q; for i in ${x-aq${x-"bqc"}qd}; do echo "z${i}z"; done' \
304 1.2 christos 'zaz zbqcz zdz'
305 1.9 kre
306 1.9 kre check_results ifs_alpha
307 1.1 jruoho }
308 1.1 jruoho
309 1.1 jruoho atf_test_case quote
310 1.9 kre quote_head()
311 1.9 kre {
312 1.1 jruoho atf_set "descr" "Checks that field splitting works with multi-word" \
313 1.1 jruoho "fields"
314 1.1 jruoho }
315 1.9 kre quote_body()
316 1.9 kre {
317 1.1 jruoho unset x
318 1.1 jruoho
319 1.2 christos TEST=0
320 1.1 jruoho # Some quote propagation checks
321 1.1 jruoho check 'set "${x-a b c}"; echo $#' 1
322 1.6 kre
323 1.6 kre # this is another almost certainly incorrect expectation
324 1.7 kre # (but again, see comment in default_val test - becoming unspecified.)
325 1.6 kre # qqqq uuu qqq (quoted/unquoted) $1 is a $# is 2
326 1.1 jruoho check 'set "${x-"a b" c}"; echo $1' 'a b c'
327 1.6 kre
328 1.1 jruoho check 'for i in "${x-a b c}"; do echo "z${i}z"; done' 'za b cz'
329 1.9 kre
330 1.9 kre check_results quote
331 1.1 jruoho }
332 1.1 jruoho
333 1.1 jruoho atf_test_case dollar_at
334 1.9 kre dollar_at_head()
335 1.9 kre {
336 1.1 jruoho atf_set "descr" "Checks that field splitting works when expanding" \
337 1.1 jruoho "\$@"
338 1.1 jruoho }
339 1.9 kre dollar_at_body()
340 1.9 kre {
341 1.1 jruoho unset x
342 1.1 jruoho
343 1.2 christos TEST=0
344 1.1 jruoho # Check we get "$@" right
345 1.2 christos
346 1.2 christos check 'set --; for i in x"$@"x; do echo "z${i}z"; done' 'zxxz'
347 1.2 christos check 'set a; for i in x"$@"x; do echo "z${i}z"; done' 'zxaxz'
348 1.9 kre check 'set a b; for i in x"$@"x; do echo "z${i}z"; done' \
349 1.9 kre 'zxaz zbxz'
350 1.2 christos
351 1.2 christos check 'set --; for i; do echo "z${i}z"; done' ''
352 1.2 christos check 'set --; for i in $@; do echo "z${i}z"; done' ''
353 1.2 christos check 'set --; for i in "$@"; do echo "z${i}z"; done' ''
354 1.2 christos # atf_expect_fail "PR bin/50834"
355 1.2 christos check 'set --; for i in ""$@; do echo "z${i}z"; done' 'zz'
356 1.2 christos # atf_expect_pass
357 1.2 christos check 'set --; for i in $@""; do echo "z${i}z"; done' 'zz'
358 1.2 christos check 'set --; for i in ""$@""; do echo "z${i}z"; done' 'zz'
359 1.2 christos check 'set --; for i in """$@"; do echo "z${i}z"; done' 'zz'
360 1.2 christos check 'set --; for i in "$@"""; do echo "z${i}z"; done' 'zz'
361 1.2 christos check 'set --; for i in """$@""";do echo "z${i}z"; done' 'zz'
362 1.2 christos
363 1.2 christos check 'set ""; for i; do echo "z${i}z"; done' 'zz'
364 1.2 christos check 'set ""; for i in "$@"; do echo "z${i}z"; done' 'zz'
365 1.2 christos check 'set "" ""; for i; do echo "z${i}z"; done' 'zz zz'
366 1.2 christos check 'set "" ""; for i in "$@"; do echo "z${i}z"; done' 'zz zz'
367 1.2 christos check 'set "" ""; for i in $@; do echo "z${i}z"; done' ''
368 1.2 christos
369 1.2 christos check 'set "a b" c; for i; do echo "z${i}z"; done' \
370 1.2 christos 'za bz zcz'
371 1.2 christos check 'set "a b" c; for i in "$@"; do echo "z${i}z"; done' \
372 1.2 christos 'za bz zcz'
373 1.2 christos check 'set "a b" c; for i in $@; do echo "z${i}z"; done' \
374 1.2 christos 'zaz zbz zcz'
375 1.2 christos check 'set " a b " c; for i in "$@"; do echo "z${i}z"; done' \
376 1.2 christos 'z a b z zcz'
377 1.4 christos
378 1.4 christos check 'set a b c; for i in "$@$@"; do echo "z${i}z"; done' \
379 1.4 christos 'zaz zbz zcaz zbz zcz'
380 1.4 christos check 'set a b c; for i in "$@""$@";do echo "z${i}z"; done' \
381 1.4 christos 'zaz zbz zcaz zbz zcz'
382 1.9 kre
383 1.9 kre check_results dollar_at
384 1.1 jruoho }
385 1.1 jruoho
386 1.1 jruoho atf_test_case ifs
387 1.9 kre ifs_head()
388 1.9 kre {
389 1.1 jruoho atf_set "descr" "Checks that IFS correctly configures field" \
390 1.1 jruoho "splitting behavior"
391 1.1 jruoho }
392 1.9 kre ifs_body()
393 1.9 kre {
394 1.1 jruoho unset x
395 1.1 jruoho
396 1.2 christos TEST=0
397 1.1 jruoho # Some IFS tests
398 1.9 kre check 't="-- "; IFS=" "; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '0'
399 1.9 kre check 't=" x"; IFS=" x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '1'
400 1.9 kre check 't=" x "; IFS=" x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' '1'
401 1.9 kre check 't=axb; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' \
402 1.9 kre '2 a:b'
403 1.9 kre check 't="a x b"; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' \
404 1.9 kre '2 a : b'
405 1.9 kre check 't="a xx b"; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' \
406 1.9 kre '3 a :: b'
407 1.9 kre check 't="a xx b"; IFS="x ";set $t; IFS=":"; r="$*"; IFS=; echo $# $r' \
408 1.9 kre '3 a::b'
409 1.9 kre # A recent 'clarification' means that a single trailing IFS
410 1.9 kre # non-whitespace doesn't generate an empty parameter
411 1.9 kre check 't="xax"; IFS="x"; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' \
412 1.9 kre '2 :a'
413 1.9 kre check 't="xax "; IFS="x "; set $t; IFS=":"; r="$*"; IFS=; echo $# $r' \
414 1.9 kre '2 :a'
415 1.1 jruoho # Verify that IFS isn't being applied where it shouldn't be.
416 1.9 kre check 'IFS="x"; set axb; IFS=":"; r="$*"; IFS=; echo $# $r' '1 axb'
417 1.9 kre check 'IFS=x; set axb; IFS=:; r=$*; IFS=; echo $# $r' '1 axb'
418 1.9 kre check 'IFS=x; set axb; set -- "$*"; IFS=:; r=$*; IFS=; echo $# $r' \
419 1.9 kre '1 axb'
420 1.9 kre check 'IFS=x; set axb; set -- $* ; IFS=:; r=$*; IFS=; echo $# $r' \
421 1.9 kre '2 a:b'
422 1.9 kre
423 1.9 kre check_results ifs
424 1.1 jruoho }
425 1.1 jruoho
426 1.1 jruoho atf_test_case var_length
427 1.9 kre var_length_head()
428 1.9 kre {
429 1.1 jruoho atf_set "descr" "Checks that field splitting works when expanding" \
430 1.1 jruoho "a variable's length"
431 1.1 jruoho }
432 1.9 kre var_length_body()
433 1.9 kre {
434 1.2 christos TEST=0
435 1.1 jruoho
436 1.1 jruoho long=12345678123456781234567812345678
437 1.1 jruoho long=$long$long$long$long
438 1.2 christos export long
439 1.5 kre unset x
440 1.2 christos
441 1.2 christos # first test that the test method works...
442 1.2 christos check 'set -u; : ${long}; echo ${#long}' '128'
443 1.2 christos
444 1.2 christos # Check that we apply IFS to ${#var}
445 1.2 christos check 'echo ${#long}; IFS=2; echo ${#long}; set 1 ${#long};echo $#' \
446 1.2 christos '128 1 8 3'
447 1.5 kre check 'IFS=2; set ${x-${#long}}; IFS=" "; echo $* $#' '1 8 2'
448 1.5 kre check 'IFS=2; set ${x-"${#long}"}; IFS=" "; echo $* $#' '128 1'
449 1.5 kre check 'IFS=2; set "${x-${#long}}"; IFS=" "; echo $* $#' '128 1'
450 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo $* $#' '1 8 '
451 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo $* "$#"' '1 8 2'
452 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo "$*" "$#"' '128 2'
453 1.5 kre check 'IFS=2; set ${x-${#long}}; : ; echo "$@" "$#"' '1 8 2'
454 1.9 kre
455 1.9 kre check_results var_length
456 1.1 jruoho }
457 1.1 jruoho
458 1.6 kre atf_test_case split_arith
459 1.9 kre split_arith_head()
460 1.9 kre {
461 1.6 kre atf_set "descr" "Checks that field splitting works when expanding" \
462 1.6 kre "the results from arithmetic"
463 1.6 kre }
464 1.9 kre split_arith_body()
465 1.9 kre {
466 1.6 kre TEST=0
467 1.6 kre
468 1.6 kre # Check that we apply IFS to $(( expr ))
469 1.6 kre
470 1.6 kre # Note: we do not check the actual arithmetic operations here
471 1.6 kre # (there is a separate test just for that) so we just enter
472 1.6 kre # the "answer" inside $(( )) ... also makes it easier to visualise
473 1.6 kre
474 1.6 kre check 'IFS=5; echo $(( 123456789 ))' '1234 6789'
475 1.6 kre check 'IFS=5; echo "$(( 123456789 ))"' '123456789'
476 1.6 kre check 'IFS=37; echo $(( 123456789 ))' '12 456 89'
477 1.6 kre check 'IFS=37; echo "$(( 123456789 ))"' '123456789'
478 1.6 kre check 'IFS=159; echo $(( 123456789 ))' ' 234 678'
479 1.6 kre
480 1.6 kre check 'IFS=5; set -- $(( 123456789 )); echo $#: $1 $2 $3 $4' \
481 1.6 kre '2: 1234 6789'
482 1.6 kre check 'IFS=5; set -- "$(( 123456789 ))"; echo $#: $1 $2 $3 $4' \
483 1.6 kre '1: 1234 6789' # go ahead: explain it!
484 1.6 kre check 'IFS=5; set -- "$(( 123456789 ))"; echo "$#: $1 $2 $3 $4"' \
485 1.6 kre '1: 123456789 ' # ah!
486 1.6 kre
487 1.6 kre check 'IFS=37; set -- $(( 123456789 )); echo $#: $1 $2 $3 $4' \
488 1.6 kre ' : 12 456 89' # Tricky!
489 1.6 kre check 'IFS=5; set -- $(( 123456789 )); echo $#: $*' \
490 1.6 kre '2: 1234 6789'
491 1.6 kre check 'IFS=47; set -- $(( 123456789 )); echo $#: $*' \
492 1.6 kre '3: 123 56 89'
493 1.6 kre check 'IFS=5; set -- $(( 123456789 )); echo "$#: $*"' \
494 1.6 kre '2: 123456789'
495 1.6 kre check 'IFS=37; set -- $(( 123456789 )); echo "$#: $*"' \
496 1.6 kre '3: 123456389' # [sic]
497 1.6 kre check 'IFS=5; set -- $(( 123456789 )); echo $#: $@' \
498 1.6 kre '2: 1234 6789'
499 1.6 kre check 'IFS=47; set -- $(( 123456789 )); echo $#: $@' \
500 1.6 kre '3: 123 56 89'
501 1.6 kre check 'IFS=5; set -- $(( 123456789 )); echo "$#: $@"' \
502 1.6 kre '2: 1234 6789'
503 1.6 kre check 'IFS=37; set -- $(( 123456789 )); echo "$#: $*"' \
504 1.6 kre '3: 123456389' # [sic]
505 1.6 kre
506 1.6 kre check 'IFS=1; set -- $(( 1111 )); echo "$#:" $*' '4: '
507 1.6 kre check 'IFS=" 1"; set -- $(( 1231231231 )); echo "$#: $*"' \
508 1.6 kre '4: 23 23 23'
509 1.6 kre check 'IFS="1 "; set -- $(( 1231231231 )); echo "$#: $*"' \
510 1.6 kre '4: 123123123'
511 1.6 kre
512 1.6 kre check 'IFS=5; echo 5$(( 123456789 ))5' '51234 67895'
513 1.6 kre check 'IFS=37; echo 73$(( 123456789 ))37' '7312 456 8937'
514 1.6 kre check 'IFS=159; echo 11$(( 123456789 ))95' '11 234 678 95'
515 1.6 kre check 'IFS="159 "; echo 11$(( 123456789 ))95' '11 234 678 95'
516 1.6 kre check 'IFS="159 "; echo 11$(( 11234567899 ))95' '11 234 678 95'
517 1.9 kre
518 1.9 kre check_results split_arith
519 1.6 kre }
520 1.6 kre
521 1.8 kre atf_test_case read_split
522 1.9 kre read_split_head()
523 1.9 kre {
524 1.8 kre atf_set "descr" "Checks that field splitting works for the read" \
525 1.8 kre "built-in utility"
526 1.8 kre }
527 1.8 kre #
528 1.8 kre # CAUTION: There are literal <tab> chars in the following test.
529 1.8 kre # It is important that they be retained as is (the ones in the data
530 1.8 kre # and results - those used for test formatting are immaterial).
531 1.8 kre #
532 1.9 kre read_split_body()
533 1.9 kre {
534 1.8 kre DATA=" aaa bbb:ccc ddd+eee fff:ggg+hhh " # CAUTION: tabs!
535 1.8 kre
536 1.8 kre TEST=0
537 1.8 kre
538 1.8 kre check "unset IFS; printf '%s\n' '${DATA}' | {
539 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
540 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
541 1.8 kre '<aaa><bbb:ccc><ddd+eee><fff:ggg+hhh><><><><>'
542 1.8 kre
543 1.8 kre check "unset IFS; printf '%s\n' '${DATA}' | {
544 1.8 kre read x || printf 'FAIL:%d' \"\$?\" &&
545 1.8 kre printf '<%s>' "'"$x"; }' \
546 1.8 kre '<aaa bbb:ccc ddd+eee fff:ggg+hhh>'
547 1.8 kre
548 1.8 kre check "IFS=; printf '%s\n' '${DATA}' | {
549 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
550 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
551 1.8 kre "<${DATA}><><><><><><><>"
552 1.8 kre
553 1.8 kre check "IFS=' '; printf '%s\n' '${DATA}' | {
554 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
555 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
556 1.8 kre '<aaa><bbb:ccc><ddd+eee><fff:ggg+hhh><><><><>'
557 1.8 kre
558 1.8 kre check "IFS=':'; printf '%s\n' '${DATA}' | {
559 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
560 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
561 1.8 kre '< aaa bbb><ccc ddd+eee fff><ggg+hhh ><><><><><>'
562 1.8 kre
563 1.8 kre check "IFS=': '; printf '%s\n' '${DATA}' | {
564 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
565 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
566 1.8 kre '<aaa><bbb><ccc><ddd+eee fff><ggg+hhh ><><><>'
567 1.8 kre
568 1.8 kre check "IFS=': '; printf '%s\n' '${DATA}' | {
569 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
570 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
571 1.8 kre '< aaa bbb><ccc ddd+eee><fff><ggg+hhh>< ><><><>'
572 1.8 kre
573 1.8 kre check "IFS='+'; printf '%s\n' '${DATA}' | {
574 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
575 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
576 1.8 kre '< aaa bbb:ccc ddd><eee fff:ggg><hhh ><><><><><>'
577 1.8 kre
578 1.8 kre check "IFS=' +'; printf '%s\n' '${DATA}' | {
579 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
580 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
581 1.8 kre '<aaa><bbb:ccc><ddd><eee fff:ggg><hhh ><><><>'
582 1.8 kre
583 1.8 kre check "IFS='+ '; printf '%s\n' '${DATA}' | {
584 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
585 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
586 1.8 kre '< aaa bbb:ccc ddd><eee><fff:ggg><hhh>< ><><><>'
587 1.8 kre
588 1.8 kre # This tests the bug from PR bin/57849 (which existed about 2 days)
589 1.8 kre # It also tests that a var-assign before read does not corrupt the
590 1.8 kre # value of the var in the executing shell environment
591 1.8 kre check "IFS='+'; printf '%s\n' '${DATA}' | {
592 1.8 kre IFS=: read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
593 1.8 kre printf '<%s>' "'"$IFS" "$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
594 1.8 kre '<+>< aaa bbb><ccc ddd+eee fff><ggg+hhh ><><><><><>'
595 1.8 kre
596 1.8 kre check "IFS='+'; printf '%s\n' '${DATA}' | {
597 1.8 kre IFS= read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
598 1.8 kre printf '<%s>' "'"$IFS" "$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
599 1.8 kre "<+><${DATA}><><><><><><><>"
600 1.8 kre
601 1.8 kre # This doesn't really belong here, just tests that EOF works...
602 1.8 kre # (and that read sets unused vars to '', doesn't leave them unset)
603 1.8 kre check "unset IFS; set -u;
604 1.8 kre read a b c d e f g h </dev/null || printf 'FAIL:%d' \"\$?\" &&
605 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"' \
606 1.8 kre "FAIL:1<><><><><><><><>"
607 1.8 kre
608 1.8 kre # And a similar one where EOF follows some data (which is read)
609 1.8 kre check "unset IFS; set -u; printf 'a b c' | {
610 1.8 kre read a b c d e f g h || printf 'FAIL:%d' \"\$?\" &&
611 1.8 kre printf '<%s>' "'"$a" "$b" "$c" "$d" "$e" "$f" "$g" "$h"; }' \
612 1.8 kre "FAIL:1<a><b><c><><><><><>"
613 1.9 kre
614 1.9 kre check_results read_split
615 1.8 kre }
616 1.8 kre
617 1.9 kre atf_init_test_cases()
618 1.9 kre {
619 1.1 jruoho atf_add_test_case for
620 1.1 jruoho atf_add_test_case default_val
621 1.2 christos atf_add_test_case replacement_val
622 1.1 jruoho atf_add_test_case ifs_alpha
623 1.1 jruoho atf_add_test_case quote
624 1.1 jruoho atf_add_test_case dollar_at
625 1.1 jruoho atf_add_test_case ifs
626 1.1 jruoho atf_add_test_case var_length
627 1.6 kre atf_add_test_case split_arith
628 1.8 kre atf_add_test_case read_split
629 1.1 jruoho }
630