1 ## compare_shells: dash bash-4.4 mksh zsh
2
3 #### implicit for loop
4 # This is like "for i in $@".
5 fun() {
6 for i; do
7 echo $i
8 done
9 }
10 fun 1 2 3
11 ## STDOUT:
12 1
13 2
14 3
15 ## END
16
17 #### empty for loop (has "in")
18 set -- 1 2 3
19 for i in ; do
20 echo $i
21 done
22 ## stdout-json: ""
23
24 #### for loop with invalid identifier
25 # should be compile time error, but runtime error is OK too
26 for - in a b c; do
27 echo hi
28 done
29 ## stdout-json: ""
30 ## status: 2
31 ## OK bash/mksh status: 1
32 ## BUG zsh stdout: hi
33 ## BUG zsh status: 1
34
35 #### the word 'in' can be the loop variable
36
37 for in in a b c; do
38 echo $in
39 done
40 ## STDOUT:
41 a
42 b
43 c
44 ## END
45
46 #### Tilde expansion within for loop
47 HOME=/home/bob
48 for name in ~/src ~/git; do
49 echo $name
50 done
51 ## STDOUT:
52 /home/bob/src
53 /home/bob/git
54 ## END
55
56 #### Brace Expansion within Array
57 for i in -{a,b} {c,d}-; do
58 echo $i
59 done
60 ## STDOUT:
61 -a
62 -b
63 c-
64 d-
65 ## END
66 ## N-I dash STDOUT:
67 -{a,b}
68 {c,d}-
69 ## END
70
71 #### using loop var outside loop
72 fun() {
73 for i in a b c; do
74 echo $i
75 done
76 echo $i
77 }
78 fun
79 ## status: 0
80 ## STDOUT:
81 a
82 b
83 c
84 c
85 ## END
86
87 #### continue
88 for i in a b c; do
89 echo $i
90 if test $i = b; then
91 continue
92 fi
93 echo $i
94 done
95 ## status: 0
96 ## STDOUT:
97 a
98 a
99 b
100 c
101 c
102 ## END
103
104 #### break
105 for i in a b c; do
106 echo $i
107 if test $i = b; then
108 break
109 fi
110 done
111 ## status: 0
112 ## STDOUT:
113 a
114 b
115 ## END
116
117 #### dynamic control flow (KNOWN INCOMPATIBILITY)
118 # hm would it be saner to make FATAL builtins called break/continue/etc.?
119 # On the other hand, this spits out errors loudly.
120 b=break
121 for i in 1 2 3; do
122 echo $i
123 $b
124 done
125 ## STDOUT:
126 1
127 ## END
128 ## OK osh STDOUT:
129 1
130 2
131 3
132 ## END
133 ## OK osh status: 127
134
135 #### while in while condition
136 # This is a consequence of the grammar
137 while while true; do echo cond; break; done
138 do
139 echo body
140 break
141 done
142 ## STDOUT:
143 cond
144 body
145 ## END
146
147 #### while in pipe
148 x=$(find spec/ | wc -l)
149 y=$(find spec/ | while read path; do
150 echo $path
151 done | wc -l
152 )
153 test $x -eq $y
154 echo status=$?
155 ## stdout: status=0
156
157 #### while in pipe with subshell
158 i=0
159 seq 3 | ( while read foo; do
160 i=$((i+1))
161 #echo $i
162 done
163 echo $i )
164 ## stdout: 3
165
166 #### until loop
167 # This is just the opposite of while? while ! cond?
168 until false; do
169 echo hi
170 break
171 done
172 ## stdout: hi
173
174 #### continue at top level
175 if true; then
176 echo one
177 continue
178 echo two
179 fi
180 ## status: 0
181 ## STDOUT:
182 one
183 two
184 ## END
185 # zsh behaves like strict_control_flow!
186 ## OK zsh status: 1
187 ## OK zsh STDOUT:
188 one
189 ## END
190
191 #### continue in subshell
192 for i in $(seq 2); do
193 echo "> $i"
194 ( if true; then continue; fi; echo "Should not print" )
195 echo subshell status=$?
196 echo ". $i"
197 done
198 ## STDOUT:
199 # osh lets you fail
200 > 1
201 subshell status=1
202 . 1
203 > 2
204 subshell status=1
205 . 2
206 ## END
207 ## OK dash/bash/zsh STDOUT:
208 > 1
209 subshell status=0
210 . 1
211 > 2
212 subshell status=0
213 . 2
214 ## END
215 ## BUG mksh STDOUT:
216 > 1
217 Should not print
218 subshell status=0
219 . 1
220 > 2
221 Should not print
222 subshell status=0
223 . 2
224 ## END
225
226 #### continue in subshell aborts with errexit
227 # The other shells don't let you recover from this programming error!
228 set -o errexit
229 for i in $(seq 2); do
230 echo "> $i"
231 ( if true; then continue; fi; echo "Should not print" )
232 echo 'should fail after subshell'
233 echo ". $i"
234 done
235 ## STDOUT:
236 > 1
237 ## END
238 ## status: 1
239 ## BUG dash/bash/zsh STDOUT:
240 > 1
241 should fail after subshell
242 . 1
243 > 2
244 should fail after subshell
245 . 2
246 ## END
247 ## BUG dash/bash/zsh status: 0
248 ## BUG mksh STDOUT:
249 > 1
250 Should not print
251 should fail after subshell
252 . 1
253 > 2
254 Should not print
255 should fail after subshell
256 . 2
257 ## END
258 ## BUG mksh status: 0
259
260 #### bad arg to break
261 x=oops
262 while true; do
263 echo hi
264 break $x
265 sleep 0.1
266 done
267 ## stdout: hi
268 ## status: 1
269 ## OK dash status: 2
270 ## OK bash status: 128
271
272 #### too many args to continue
273 # OSH treats this as a parse error
274 for x in a b c; do
275 echo $x
276 # bash breaks rather than continue or fatal error!!!
277 continue 1 2 3
278 done
279 echo --
280 ## stdout-json: ""
281 ## status: 2
282 ## BUG bash STDOUT:
283 a
284 --
285 ## END
286 ## BUG bash status: 0
287 ## BUG dash/mksh/zsh STDOUT:
288 a
289 b
290 c
291 --
292 ## END
293 ## BUG dash/mksh/zsh status: 0
294
295 #### break in condition of loop
296 while break; do
297 echo x
298 done
299 echo done
300 ## STDOUT:
301 done
302 ## END
303
304
305 #### break in condition of nested loop
306 for i in 1 2 3; do
307 echo i=$i
308 while break; do
309 echo x
310 done
311 done
312 echo done
313 ## STDOUT:
314 i=1
315 i=2
316 i=3
317 done
318 ## END
319
320 #### return within eval
321 f() {
322 echo one
323 eval 'return'
324 echo two
325 }
326 f
327 ## STDOUT:
328 one
329 ## END
330
331 #### break/continue within eval
332 # NOTE: This changes things
333 # set -e
334 f() {
335 for i in $(seq 5); do
336 if test $i = 2; then
337 eval continue
338 fi
339 if test $i = 4; then
340 eval break
341 fi
342 echo $i
343 done
344
345 eval 'return'
346 echo 'done'
347 }
348 f
349 ## STDOUT:
350 1
351 3
352 ## END
353 ## BUG mksh STDOUT:
354 1
355 2
356 3
357 4
358 5
359 ## END
360
361 #### break/continue within source
362 # NOTE: This changes things
363 # set -e
364
365 cd $REPO_ROOT
366 f() {
367 for i in $(seq 5); do
368 if test $i = 2; then
369 . spec/testdata/continue.sh
370 fi
371 if test $i = 4; then
372 . spec/testdata/break.sh
373 fi
374 echo $i
375 done
376
377 # Return is different!
378 . spec/testdata/return.sh
379 echo done
380 }
381 f
382 ## STDOUT:
383 1
384 3
385 done
386 ## END
387 ## BUG zsh/mksh STDOUT:
388 1
389 2
390 3
391 4
392 5
393 done
394 ## END
395
396 #### top-level break/continue/return (without strict_control_flow)
397 $SH -c 'break; echo break=$?'
398 $SH -c 'continue; echo continue=$?'
399 $SH -c 'return; echo return=$?'
400 ## STDOUT:
401 break=0
402 continue=0
403 ## END
404 ## BUG zsh stdout-json: ""
405 ## BUG bash STDOUT:
406 break=0
407 continue=0
408 return=1
409 ## END
410
411
412 #### multi-level break with argument
413
414 # reported in issue #1459
415
416 counterA=100
417 counterB=100
418
419 while test "$counterA" -gt 0
420 do
421 counterA=$((counterA - 1))
422 while test "$counterB" -gt 0
423 do
424 counterB=$((counterB - 1))
425 if test "$counterB" = 50
426 then
427 break 2
428 fi
429 done
430 done
431
432 echo "$counterA"
433 echo "$counterB"
434
435 ## STDOUT:
436 99
437 50
438 ## END
439
440
441 #### multi-level continue
442
443 for i in 1 2; do
444 for j in a b c; do
445 if test $j = b; then
446 continue
447 fi
448 echo $i $j
449 done
450 done
451
452 echo ---
453
454 for i in 1 2; do
455 for j in a b c; do
456 if test $j = b; then
457 continue 2 # MULTI-LEVEL
458 fi
459 echo $i $j
460 done
461 done
462
463
464 ## STDOUT:
465 1 a
466 1 c
467 2 a
468 2 c
469 ---
470 1 a
471 2 a
472 ## END
473
474