1 | ## compare_shells: bash dash mksh
|
2 | ## oils_failures_allowed: 9
|
3 |
|
4 | # NOTE on bash bug: After setting IFS to array, it never splits anymore? Even
|
5 | # if you assign IFS again.
|
6 |
|
7 | #### IFS is scoped
|
8 | IFS=b
|
9 | word=abcd
|
10 | f() { local IFS=c; argv.py $word; }
|
11 | f
|
12 | argv.py $word
|
13 | ## stdout-json: "['ab', 'd']\n['a', 'cd']\n"
|
14 |
|
15 | #### Tilde sub is not split, but var sub is
|
16 | HOME="foo bar"
|
17 | argv.py ~
|
18 | argv.py $HOME
|
19 | ## stdout-json: "['foo bar']\n['foo', 'bar']\n"
|
20 |
|
21 | #### Word splitting
|
22 | a="1 2"
|
23 | b="3 4"
|
24 | argv.py $a"$b"
|
25 | ## stdout-json: "['1', '23 4']\n"
|
26 |
|
27 | #### Word splitting 2
|
28 | a="1 2"
|
29 | b="3 4"
|
30 | c="5 6"
|
31 | d="7 8"
|
32 | argv.py $a"$b"$c"$d"
|
33 | ## stdout-json: "['1', '23 45', '67 8']\n"
|
34 |
|
35 | # Has tests on differences between $* "$*" $@ "$@"
|
36 | # http://stackoverflow.com/questions/448407/bash-script-to-receive-and-repass-quoted-parameters
|
37 |
|
38 | #### $*
|
39 | fun() { argv.py -$*-; }
|
40 | fun "a 1" "b 2" "c 3"
|
41 | ## stdout: ['-a', '1', 'b', '2', 'c', '3-']
|
42 |
|
43 | #### "$*"
|
44 | fun() { argv.py "-$*-"; }
|
45 | fun "a 1" "b 2" "c 3"
|
46 | ## stdout: ['-a 1 b 2 c 3-']
|
47 |
|
48 | #### $@
|
49 | # How does this differ from $* ? I don't think it does.
|
50 | fun() { argv.py -$@-; }
|
51 | fun "a 1" "b 2" "c 3"
|
52 | ## stdout: ['-a', '1', 'b', '2', 'c', '3-']
|
53 |
|
54 | #### "$@"
|
55 | fun() { argv.py "-$@-"; }
|
56 | fun "a 1" "b 2" "c 3"
|
57 | ## stdout: ['-a 1', 'b 2', 'c 3-']
|
58 |
|
59 | #### empty argv
|
60 | argv.py 1 "$@" 2 $@ 3 "$*" 4 $* 5
|
61 | ## stdout: ['1', '2', '3', '', '4', '5']
|
62 |
|
63 | #### Word elision with space
|
64 | s1=' '
|
65 | argv.py $s1
|
66 | ## stdout: []
|
67 |
|
68 | #### Word elision with non-whitespace IFS
|
69 | # Treated differently than the default IFS. What is the rule here?
|
70 | IFS='_'
|
71 | char='_'
|
72 | space=' '
|
73 | empty=''
|
74 | argv.py $char
|
75 | argv.py $space
|
76 | argv.py $empty
|
77 | ## STDOUT:
|
78 | ['']
|
79 | [' ']
|
80 | []
|
81 | ## END
|
82 |
|
83 | #### Leading/trailing word elision with non-whitespace IFS
|
84 | # This behavior is weird.
|
85 | IFS=_
|
86 | s1='_a_b_'
|
87 | argv.py $s1
|
88 | ## stdout: ['', 'a', 'b']
|
89 |
|
90 | #### Leading ' ' vs leading ' _ '
|
91 | # This behavior is weird, but all shells agree.
|
92 | IFS='_ '
|
93 | s1='_ a b _ '
|
94 | s2=' a b _ '
|
95 | argv.py $s1
|
96 | argv.py $s2
|
97 | ## STDOUT:
|
98 | ['', 'a', 'b']
|
99 | ['a', 'b']
|
100 | ## END
|
101 |
|
102 | #### Multiple non-whitespace IFS chars.
|
103 | IFS=_-
|
104 | s1='a__b---c_d'
|
105 | argv.py $s1
|
106 | ## stdout: ['a', '', 'b', '', '', 'c', 'd']
|
107 |
|
108 | #### IFS with whitespace and non-whitepace.
|
109 | # NOTE: Three delimiters means two empty words in the middle. No elision.
|
110 | IFS='_ '
|
111 | s1='a_b _ _ _ c _d e'
|
112 | argv.py $s1
|
113 | ## stdout: ['a', 'b', '', '', 'c', 'd', 'e']
|
114 |
|
115 | #### empty $@ and $* is elided
|
116 | fun() { argv.py 1 $@ $* 2; }
|
117 | fun
|
118 | ## stdout: ['1', '2']
|
119 |
|
120 | #### unquoted empty arg is elided
|
121 | empty=""
|
122 | argv.py 1 $empty 2
|
123 | ## stdout: ['1', '2']
|
124 |
|
125 | #### unquoted whitespace arg is elided
|
126 | space=" "
|
127 | argv.py 1 $space 2
|
128 | ## stdout: ['1', '2']
|
129 |
|
130 | #### empty literals are not elided
|
131 | space=" "
|
132 | argv.py 1 $space"" 2
|
133 | ## stdout: ['1', '', '2']
|
134 |
|
135 | #### no splitting when IFS is empty
|
136 | IFS=""
|
137 | foo="a b"
|
138 | argv.py $foo
|
139 | ## stdout: ['a b']
|
140 |
|
141 | #### default value can yield multiple words
|
142 | argv.py 1 ${undefined:-"2 3" "4 5"} 6
|
143 | ## stdout: ['1', '2 3', '4 5', '6']
|
144 |
|
145 | #### default value can yield multiple words with part joining
|
146 | argv.py 1${undefined:-"2 3" "4 5"}6
|
147 | ## stdout: ['12 3', '4 56']
|
148 |
|
149 | #### default value with unquoted IFS char
|
150 | IFS=_
|
151 | argv.py 1${undefined:-"2_3"x_x"4_5"}6
|
152 | ## stdout: ['12_3x', 'x4_56']
|
153 |
|
154 | #### IFS empty doesn't do splitting
|
155 | IFS=''
|
156 | x=$(echo -e ' a b\tc\n')
|
157 | argv.py $x
|
158 | ## STDOUT:
|
159 | [' a b\tc']
|
160 | ## END
|
161 | ## N-I dash STDOUT:
|
162 | ['-e a b\tc']
|
163 | ## END
|
164 |
|
165 |
|
166 | #### IFS unset behaves like $' \t\n'
|
167 | unset IFS
|
168 | x=$(echo -e ' a b\tc\n')
|
169 | argv.py $x
|
170 | ## STDOUT:
|
171 | ['a', 'b', 'c']
|
172 | ## END
|
173 | ## N-I dash STDOUT:
|
174 | ['-e', 'a', 'b', 'c']
|
175 | ## END
|
176 |
|
177 | #### IFS='\'
|
178 | # NOTE: OSH fails this because of double backslash escaping issue!
|
179 | IFS='\'
|
180 | s='a\b'
|
181 | argv.py $s
|
182 | ## STDOUT:
|
183 | ['a', 'b']
|
184 | ## END
|
185 |
|
186 | #### IFS='\ '
|
187 | # NOTE: OSH fails this because of double backslash escaping issue!
|
188 | # When IFS is \, then you're no longer using backslash escaping.
|
189 | IFS='\ '
|
190 | s='a\b \\ c d\'
|
191 | argv.py $s
|
192 | ## STDOUT:
|
193 | ['a', 'b', '', 'c', 'd']
|
194 | ## END
|
195 |
|
196 | #### IFS characters are glob metacharacters
|
197 | IFS='* '
|
198 | s='a*b c'
|
199 | argv.py $s
|
200 |
|
201 | IFS='?'
|
202 | s='?x?y?z?'
|
203 | argv.py $s
|
204 |
|
205 | IFS='['
|
206 | s='[x[y[z['
|
207 | argv.py $s
|
208 | ## STDOUT:
|
209 | ['a', 'b', 'c']
|
210 | ['', 'x', 'y', 'z']
|
211 | ['', 'x', 'y', 'z']
|
212 | ## END
|
213 |
|
214 | #### Trailing space
|
215 | argv.py 'Xec ho '
|
216 | argv.py X'ec ho '
|
217 | argv.py X"ec ho "
|
218 | ## STDOUT:
|
219 | ['Xec ho ']
|
220 | ['Xec ho ']
|
221 | ['Xec ho ']
|
222 | ## END
|
223 |
|
224 | #### Empty IFS (regression for bug)
|
225 | IFS=
|
226 | echo ["$*"]
|
227 | set a b c
|
228 | echo ["$*"]
|
229 | ## STDOUT:
|
230 | []
|
231 | [abc]
|
232 | ## END
|
233 |
|
234 | #### Unset IFS (regression for bug)
|
235 | set a b c
|
236 | unset IFS
|
237 | echo ["$*"]
|
238 | ## STDOUT:
|
239 | [a b c]
|
240 | ## END
|
241 |
|
242 | #### IFS=o (regression for bug)
|
243 | IFS=o
|
244 | echo hi
|
245 | ## STDOUT:
|
246 | hi
|
247 | ## END
|
248 |
|
249 | #### IFS and joining arrays
|
250 | IFS=:
|
251 | set -- x 'y z'
|
252 | argv.py "$@"
|
253 | argv.py $@
|
254 | argv.py "$*"
|
255 | argv.py $*
|
256 | ## STDOUT:
|
257 | ['x', 'y z']
|
258 | ['x', 'y z']
|
259 | ['x:y z']
|
260 | ['x', 'y z']
|
261 | ## END
|
262 |
|
263 | #### IFS and joining arrays by assignments
|
264 | IFS=:
|
265 | set -- x 'y z'
|
266 |
|
267 | s="$@"
|
268 | argv.py "$s"
|
269 |
|
270 | s=$@
|
271 | argv.py "$s"
|
272 |
|
273 | s"$*"
|
274 | argv.py "$s"
|
275 |
|
276 | s=$*
|
277 | argv.py "$s"
|
278 |
|
279 | # bash and mksh agree, but this doesn't really make sense to me.
|
280 | # In OSH, "$@" is the only real array, so that's why it behaves differently.
|
281 |
|
282 | ## STDOUT:
|
283 | ['x y z']
|
284 | ['x y z']
|
285 | ['x y z']
|
286 | ['x:y z']
|
287 | ## END
|
288 | ## OK dash STDOUT:
|
289 | ['x:y z']
|
290 | ['x:y z']
|
291 | ['x:y z']
|
292 | ['x:y z']
|
293 | ## END
|
294 |
|
295 |
|
296 | # TODO:
|
297 | # - unquoted args of whitespace are not elided (when IFS = null)
|
298 | # - empty quoted args are kept
|
299 | #
|
300 | # - $* $@ with empty IFS
|
301 | # - $* $@ with custom IFS
|
302 | #
|
303 | # - no splitting when IFS is empty
|
304 | # - word splitting removes leading and trailing whitespace
|
305 |
|
306 | # TODO: test framework needs common setup
|
307 |
|
308 | # Test IFS and $@ $* on all these
|
309 | #### TODO
|
310 | empty=""
|
311 | space=" "
|
312 | AB="A B"
|
313 | X="X"
|
314 | Yspaces=" Y "
|
315 |
|
316 |
|
317 | #### IFS='' with $@ and $* (bug #627)
|
318 | set -- a 'b c'
|
319 | IFS=''
|
320 | argv.py at $@
|
321 | argv.py star $*
|
322 |
|
323 | # zsh agrees
|
324 | ## STDOUT:
|
325 | ['at', 'a', 'b c']
|
326 | ['star', 'a', 'b c']
|
327 | ## END
|
328 | ## BUG ash STDOUT:
|
329 | ['at', 'ab c']
|
330 | ['star', 'ab c']
|
331 | ## END
|
332 |
|
333 | #### IFS='' with $@ and $* and printf (bug #627)
|
334 | set -- a 'b c'
|
335 | IFS=''
|
336 | printf '[%s]\n' $@
|
337 | printf '[%s]\n' $*
|
338 | ## STDOUT:
|
339 | [a]
|
340 | [b c]
|
341 | [a]
|
342 | [b c]
|
343 | ## END
|
344 | ## BUG ash STDOUT:
|
345 | [ab c]
|
346 | [ab c]
|
347 | ## END
|
348 |
|
349 | #### IFS='' with ${a[@]} and ${a[*]} (bug #627)
|
350 | myarray=(a 'b c')
|
351 | IFS=''
|
352 | argv.py at ${myarray[@]}
|
353 | argv.py star ${myarray[*]}
|
354 |
|
355 | ## STDOUT:
|
356 | ['at', 'a', 'b c']
|
357 | ['star', 'a', 'b c']
|
358 | ## END
|
359 | ## N-I dash/ash status: 2
|
360 | ## N-I dash/ash stdout-json: ""
|
361 |
|
362 | #### Bug #628 split on : with : in literal word
|
363 | IFS=':'
|
364 | word='a:'
|
365 | argv.py ${word}:b
|
366 | argv.py ${word}:
|
367 |
|
368 | echo ---
|
369 |
|
370 | # Same thing happens for 'z'
|
371 | IFS='z'
|
372 | word='az'
|
373 | argv.py ${word}zb
|
374 | argv.py ${word}z
|
375 | ## STDOUT:
|
376 | ['a', ':b']
|
377 | ['a', ':']
|
378 | ---
|
379 | ['a', 'zb']
|
380 | ['a', 'z']
|
381 | ## END
|
382 |
|
383 | #### Bug #698, similar crash
|
384 | var='\'
|
385 | set -f
|
386 | echo $var
|
387 | ## STDOUT:
|
388 | \
|
389 | ## END
|
390 |
|
391 | #### Bug #1664, \\ with noglob
|
392 |
|
393 | # Note that we're not changing IFS
|
394 |
|
395 | argv.py [\\]_
|
396 | argv.py "[\\]_"
|
397 |
|
398 | # TODO: no difference observed here, go back to original bug
|
399 |
|
400 | #argv.py [\\_
|
401 | #argv.py "[\\_"
|
402 |
|
403 | echo noglob
|
404 |
|
405 | # repeat cases with -f, noglob
|
406 | set -f
|
407 |
|
408 | argv.py [\\]_
|
409 | argv.py "[\\]_"
|
410 |
|
411 | #argv.py [\\_
|
412 | #argv.py "[\\_"
|
413 |
|
414 | ## STDOUT:
|
415 | ['[\\]_']
|
416 | ['[\\]_']
|
417 | noglob
|
418 | ['[\\]_']
|
419 | ['[\\]_']
|
420 | ## END
|
421 |
|
422 |
|
423 | #### Empty IFS bug #2141 (from pnut)
|
424 |
|
425 | res=0
|
426 | sum() {
|
427 | # implement callee-save calling convention using `set`
|
428 | # here, we save the value of $res after the function parameters
|
429 | set $@ $res # $1 $2 $3 are now set
|
430 | res=$(($1 + $2))
|
431 | echo "$1 + $2 = $res"
|
432 | res=$3 # restore the value of $res
|
433 | }
|
434 |
|
435 | unset IFS
|
436 | sum 12 30 # outputs "12 + 30 = 42"
|
437 |
|
438 | IFS=' '
|
439 | sum 12 30 # outputs "12 + 30 = 42"
|
440 |
|
441 | IFS=
|
442 | sum 12 30 # outputs "1230 + 0 = 1230"
|
443 |
|
444 | # I added this
|
445 | IFS=''
|
446 | sum 12 30
|
447 |
|
448 | set -u
|
449 | IFS=
|
450 | sum 12 30 # fails with "fatal: Undefined variable '2'" on res=$(($1 + $2))
|
451 |
|
452 | ## STDOUT:
|
453 | 12 + 30 = 42
|
454 | 12 + 30 = 42
|
455 | 12 + 30 = 42
|
456 | 12 + 30 = 42
|
457 | 12 + 30 = 42
|
458 | ## END
|