1 | ## compare_shells: bash dash mksh zsh
|
2 |
|
3 | #### Lazy Evaluation of Alternative
|
4 | i=0
|
5 | x=x
|
6 | echo ${x:-$((i++))}
|
7 | echo $i
|
8 | echo ${undefined:-$((i++))}
|
9 | echo $i # i is one because the alternative was only evaluated once
|
10 | ## status: 0
|
11 | ## stdout-json: "x\n0\n0\n1\n"
|
12 | ## N-I dash status: 2
|
13 | ## N-I dash stdout-json: "x\n0\n"
|
14 |
|
15 | #### Default value when empty
|
16 | empty=''
|
17 | echo ${empty:-is empty}
|
18 | ## stdout: is empty
|
19 |
|
20 | #### Default value when unset
|
21 | echo ${unset-is unset}
|
22 | ## stdout: is unset
|
23 |
|
24 | #### Unquoted with array as default value
|
25 | set -- '1 2' '3 4'
|
26 | argv.py X${unset=x"$@"x}X
|
27 | argv.py X${unset=x$@x}X # If you want OSH to split, write this
|
28 | # osh
|
29 | ## STDOUT:
|
30 | ['Xx1', '2', '3', '4xX']
|
31 | ['Xx1', '2', '3', '4xX']
|
32 | ## END
|
33 | ## OK osh STDOUT:
|
34 | ['Xx1 2', '3 4xX']
|
35 | ['Xx1', '2', '3', '4xX']
|
36 | ## END
|
37 | ## OK zsh STDOUT:
|
38 | ['Xx1 2 3 4xX']
|
39 | ['Xx1 2 3 4xX']
|
40 | ## END
|
41 |
|
42 | #### Quoted with array as default value
|
43 | set -- '1 2' '3 4'
|
44 | argv.py "X${unset=x"$@"x}X"
|
45 | argv.py "X${unset=x$@x}X" # OSH is the same here
|
46 | ## STDOUT:
|
47 | ['Xx1 2 3 4xX']
|
48 | ['Xx1 2 3 4xX']
|
49 | ## END
|
50 | ## BUG bash STDOUT:
|
51 | ['Xx1', '2', '3', '4xX']
|
52 | ['Xx1 2 3 4xX']
|
53 | ## END
|
54 | ## OK osh STDOUT:
|
55 | ['Xx1 2', '3 4xX']
|
56 | ['Xx1 2 3 4xX']
|
57 | ## END
|
58 |
|
59 | #### Assign default with array
|
60 | set -- '1 2' '3 4'
|
61 | argv.py X${unset=x"$@"x}X
|
62 | argv.py "$unset"
|
63 | ## STDOUT:
|
64 | ['Xx1', '2', '3', '4xX']
|
65 | ['x1 2 3 4x']
|
66 | ## END
|
67 | ## OK osh STDOUT:
|
68 | ['Xx1 2', '3 4xX']
|
69 | ['x1 2 3 4x']
|
70 | ## END
|
71 | ## OK zsh STDOUT:
|
72 | ['Xx1 2 3 4xX']
|
73 | ['x1 2 3 4x']
|
74 | ## END
|
75 |
|
76 | #### Assign default value when empty
|
77 | empty=''
|
78 | ${empty:=is empty}
|
79 | echo $empty
|
80 | ## stdout: is empty
|
81 |
|
82 | #### Assign default value when unset
|
83 | ${unset=is unset}
|
84 | echo $unset
|
85 | ## stdout: is unset
|
86 |
|
87 | #### ${v:+foo} Alternative value when empty
|
88 | v=foo
|
89 | empty=''
|
90 | echo ${v:+v is not empty} ${empty:+is not empty}
|
91 | ## stdout: v is not empty
|
92 |
|
93 | #### ${v+foo} Alternative value when unset
|
94 | v=foo
|
95 | echo ${v+v is not unset} ${unset:+is not unset}
|
96 | ## stdout: v is not unset
|
97 |
|
98 | #### "${x+foo}" quoted (regression)
|
99 | # Python's configure caught this
|
100 | argv.py "${with_icc+set}" = set
|
101 | ## STDOUT:
|
102 | ['', '=', 'set']
|
103 | ## END
|
104 |
|
105 | #### ${s+foo} and ${s:+foo} when set -u
|
106 | set -u
|
107 | v=v
|
108 | echo v=${v:+foo}
|
109 | echo v=${v+foo}
|
110 | unset v
|
111 | echo v=${v:+foo}
|
112 | echo v=${v+foo}
|
113 | ## STDOUT:
|
114 | v=foo
|
115 | v=foo
|
116 | v=
|
117 | v=
|
118 | ## END
|
119 |
|
120 | #### "${array[@]} with set -u (bash is outlier)
|
121 | case $SH in dash) exit ;; esac
|
122 |
|
123 | set -u
|
124 |
|
125 | typeset -a empty
|
126 | empty=()
|
127 |
|
128 | echo empty /"${empty[@]}"/
|
129 | echo undefined /"${undefined[@]}"/
|
130 |
|
131 | ## status: 1
|
132 | ## STDOUT:
|
133 | empty //
|
134 | ## END
|
135 |
|
136 | ## BUG bash status: 0
|
137 | ## BUG bash STDOUT:
|
138 | empty //
|
139 | undefined //
|
140 | ## END
|
141 |
|
142 | # empty array is unset in mksh
|
143 | ## BUG mksh status: 1
|
144 | ## BUG mksh STDOUT:
|
145 | ## END
|
146 |
|
147 | ## N-I dash status: 0
|
148 | ## N-I dash STDOUT:
|
149 | ## END
|
150 |
|
151 |
|
152 | #### "${undefined[@]+foo}" and "${undefined[@]:+foo}", with set -u
|
153 | case $SH in dash) exit ;; esac
|
154 |
|
155 | set -u
|
156 |
|
157 | echo plus /"${array[@]+foo}"/
|
158 | echo plus colon /"${array[@]:+foo}"/
|
159 |
|
160 | ## STDOUT:
|
161 | plus //
|
162 | plus colon //
|
163 | ## END
|
164 |
|
165 | ## N-I dash STDOUT:
|
166 | ## END
|
167 |
|
168 | #### "${a[@]+foo}" and "${a[@]:+foo}" - operators are equivalent on arrays?
|
169 |
|
170 | case $SH in dash) exit ;; esac
|
171 |
|
172 | echo '+ ' /"${array[@]+foo}"/
|
173 | echo '+:' /"${array[@]:+foo}"/
|
174 | echo
|
175 |
|
176 | typeset -a array
|
177 | array=()
|
178 |
|
179 | echo '+ ' /"${array[@]+foo}"/
|
180 | echo '+:' /"${array[@]:+foo}"/
|
181 | echo
|
182 |
|
183 | array=('')
|
184 |
|
185 | echo '+ ' /"${array[@]+foo}"/
|
186 | echo '+:' /"${array[@]:+foo}"/
|
187 | echo
|
188 |
|
189 | array=(spam eggs)
|
190 |
|
191 | echo '+ ' /"${array[@]+foo}"/
|
192 | echo '+:' /"${array[@]:+foo}"/
|
193 | echo
|
194 |
|
195 |
|
196 | ## STDOUT:
|
197 | + //
|
198 | +: //
|
199 |
|
200 | + //
|
201 | +: //
|
202 |
|
203 | + /foo/
|
204 | +: /foo/
|
205 |
|
206 | + /foo/
|
207 | +: /foo/
|
208 |
|
209 | ## END
|
210 |
|
211 | ## BUG mksh STDOUT:
|
212 | + //
|
213 | +: //
|
214 |
|
215 | + //
|
216 | +: //
|
217 |
|
218 | + /foo/
|
219 | +: //
|
220 |
|
221 | + /foo/
|
222 | +: /foo/
|
223 |
|
224 | ## END
|
225 |
|
226 | ## BUG zsh STDOUT:
|
227 | + //
|
228 | +: //
|
229 |
|
230 | + /foo/
|
231 | +: //
|
232 |
|
233 | + /foo/
|
234 | +: /foo/
|
235 |
|
236 | + /foo/
|
237 | +: /foo/
|
238 |
|
239 | ## END
|
240 |
|
241 | ## N-I dash STDOUT:
|
242 | ## END
|
243 |
|
244 |
|
245 |
|
246 | #### Nix idiom ${!hooksSlice+"${!hooksSlice}"} - was workaround for obsolete bash 4.3 bug
|
247 |
|
248 | case $SH in dash|mksh|zsh) exit ;; esac
|
249 |
|
250 | # https://oilshell.zulipchat.com/#narrow/stream/307442-nix/topic/Replacing.20bash.20with.20osh.20in.20Nixpkgs.20stdenv
|
251 |
|
252 | argv.py ${!hooksSlice+"${!hooksSlice}"}
|
253 |
|
254 | declare -a hookSlice=()
|
255 |
|
256 | argv.py ${!hooksSlice+"${!hooksSlice}"}
|
257 |
|
258 | foo=42
|
259 | bar=43
|
260 |
|
261 | declare -a hookSlice=(foo bar spam eggs)
|
262 |
|
263 | argv.py ${!hooksSlice+"${!hooksSlice}"}
|
264 |
|
265 | ## STDOUT:
|
266 | []
|
267 | []
|
268 | []
|
269 | ## END
|
270 |
|
271 | ## OK dash/mksh/zsh STDOUT:
|
272 | ## END
|
273 |
|
274 | #### ${v-foo} and ${v:-foo} when set -u
|
275 | set -u
|
276 | v=v
|
277 | echo v=${v:-foo}
|
278 | echo v=${v-foo}
|
279 | unset v
|
280 | echo v=${v:-foo}
|
281 | echo v=${v-foo}
|
282 | ## STDOUT:
|
283 | v=v
|
284 | v=v
|
285 | v=foo
|
286 | v=foo
|
287 | ## END
|
288 |
|
289 | #### array and - and +
|
290 | case $SH in (dash) exit ;; esac
|
291 |
|
292 | shopt -s compat_array # to refer to array as scalar
|
293 |
|
294 | empty=()
|
295 | a1=('')
|
296 | a2=('' x)
|
297 | a3=(3 4)
|
298 | echo empty=${empty[@]-minus}
|
299 | echo a1=${a1[@]-minus}
|
300 | echo a1[0]=${a1[0]-minus}
|
301 | echo a2=${a2[@]-minus}
|
302 | echo a3=${a3[@]-minus}
|
303 | echo ---
|
304 |
|
305 | echo empty=${empty[@]+plus}
|
306 | echo a1=${a1[@]+plus}
|
307 | echo a1[0]=${a1[0]+plus}
|
308 | echo a2=${a2[@]+plus}
|
309 | echo a3=${a3[@]+plus}
|
310 | echo ---
|
311 |
|
312 | echo empty=${empty+plus}
|
313 | echo a1=${a1+plus}
|
314 | echo a2=${a2+plus}
|
315 | echo a3=${a3+plus}
|
316 | echo ---
|
317 |
|
318 | # Test quoted arrays too
|
319 | argv.py "${empty[@]-minus}"
|
320 | argv.py "${empty[@]+plus}"
|
321 | argv.py "${a1[@]-minus}"
|
322 | argv.py "${a1[@]+plus}"
|
323 | argv.py "${a1[0]-minus}"
|
324 | argv.py "${a1[0]+plus}"
|
325 | argv.py "${a2[@]-minus}"
|
326 | argv.py "${a2[@]+plus}"
|
327 | argv.py "${a3[@]-minus}"
|
328 | argv.py "${a3[@]+plus}"
|
329 |
|
330 | ## STDOUT:
|
331 | empty=minus
|
332 | a1=
|
333 | a1[0]=
|
334 | a2= x
|
335 | a3=3 4
|
336 | ---
|
337 | empty=
|
338 | a1=plus
|
339 | a1[0]=plus
|
340 | a2=plus
|
341 | a3=plus
|
342 | ---
|
343 | empty=
|
344 | a1=plus
|
345 | a2=plus
|
346 | a3=plus
|
347 | ---
|
348 | ['minus']
|
349 | []
|
350 | ['']
|
351 | ['plus']
|
352 | ['']
|
353 | ['plus']
|
354 | ['', 'x']
|
355 | ['plus']
|
356 | ['3', '4']
|
357 | ['plus']
|
358 | ## END
|
359 | ## N-I dash stdout-json: ""
|
360 | ## N-I zsh stdout-json: "empty=\na1=\n"
|
361 | ## N-I zsh status: 1
|
362 |
|
363 | #### $@ and - and +
|
364 | echo argv=${@-minus}
|
365 | echo argv=${@+plus}
|
366 | echo argv=${@:-minus}
|
367 | echo argv=${@:+plus}
|
368 | ## STDOUT:
|
369 | argv=minus
|
370 | argv=
|
371 | argv=minus
|
372 | argv=
|
373 | ## END
|
374 | ## BUG dash/zsh STDOUT:
|
375 | argv=
|
376 | argv=plus
|
377 | argv=minus
|
378 | argv=
|
379 | ## END
|
380 |
|
381 | #### assoc array and - and +
|
382 | case $SH in (dash|mksh) exit ;; esac
|
383 |
|
384 | declare -A empty=()
|
385 | declare -A assoc=(['k']=v)
|
386 |
|
387 | echo empty=${empty[@]-minus}
|
388 | echo empty=${empty[@]+plus}
|
389 | echo assoc=${assoc[@]-minus}
|
390 | echo assoc=${assoc[@]+plus}
|
391 |
|
392 | echo ---
|
393 | echo empty=${empty[@]:-minus}
|
394 | echo empty=${empty[@]:+plus}
|
395 | echo assoc=${assoc[@]:-minus}
|
396 | echo assoc=${assoc[@]:+plus}
|
397 | ## STDOUT:
|
398 | empty=minus
|
399 | empty=
|
400 | assoc=v
|
401 | assoc=plus
|
402 | ---
|
403 | empty=minus
|
404 | empty=
|
405 | assoc=v
|
406 | assoc=plus
|
407 | ## END
|
408 |
|
409 | ## BUG zsh STDOUT:
|
410 | empty=
|
411 | empty=plus
|
412 | assoc=minus
|
413 | assoc=
|
414 | ---
|
415 | empty=minus
|
416 | empty=
|
417 | assoc=minus
|
418 | assoc=
|
419 | ## END
|
420 |
|
421 | ## N-I dash/mksh STDOUT:
|
422 | ## END
|
423 |
|
424 |
|
425 | #### Error when empty
|
426 | empty=''
|
427 | echo ${empty:?'is em'pty} # test eval of error
|
428 | echo should not get here
|
429 | ## stdout-json: ""
|
430 | ## status: 1
|
431 | ## OK dash status: 2
|
432 |
|
433 | #### Error when unset
|
434 | echo ${unset?is empty}
|
435 | echo should not get here
|
436 | ## stdout-json: ""
|
437 | ## status: 1
|
438 | ## OK dash status: 2
|
439 |
|
440 | #### Error when unset
|
441 | v=foo
|
442 | echo ${v+v is not unset} ${unset:+is not unset}
|
443 | ## stdout: v is not unset
|
444 |
|
445 | #### ${var=x} dynamic scope
|
446 | f() { : "${hello:=x}"; echo $hello; }
|
447 | f
|
448 | echo hello=$hello
|
449 |
|
450 | f() { hello=x; }
|
451 | f
|
452 | echo hello=$hello
|
453 | ## STDOUT:
|
454 | x
|
455 | hello=x
|
456 | hello=x
|
457 | ## END
|
458 |
|
459 | #### array ${arr[0]=x}
|
460 | arr=()
|
461 | echo ${#arr[@]}
|
462 | : ${arr[0]=x}
|
463 | echo ${#arr[@]}
|
464 | ## STDOUT:
|
465 | 0
|
466 | 1
|
467 | ## END
|
468 | ## N-I dash status: 2
|
469 | ## N-I dash stdout-json: ""
|
470 | ## N-I zsh status: 1
|
471 | ## N-I zsh stdout-json: "0\n"
|
472 |
|
473 | #### assoc array ${arr["k"]=x}
|
474 | # note: this also works in zsh
|
475 |
|
476 | declare -A arr=()
|
477 | echo ${#arr[@]}
|
478 | : ${arr['k']=x}
|
479 | echo ${#arr[@]}
|
480 | ## STDOUT:
|
481 | 0
|
482 | 1
|
483 | ## END
|
484 | ## N-I dash status: 2
|
485 | ## N-I dash stdout-json: ""
|
486 | ## N-I mksh status: 1
|
487 | ## N-I mksh stdout-json: ""
|
488 |
|
489 | #### "\z" as arg
|
490 | echo "${undef-\$}"
|
491 | echo "${undef-\(}"
|
492 | echo "${undef-\z}"
|
493 | echo "${undef-\"}"
|
494 | echo "${undef-\`}"
|
495 | echo "${undef-\\}"
|
496 | ## STDOUT:
|
497 | $
|
498 | \(
|
499 | \z
|
500 | "
|
501 | `
|
502 | \
|
503 | ## END
|
504 | ## BUG yash STDOUT:
|
505 | $
|
506 | (
|
507 | z
|
508 | "
|
509 | `
|
510 | \
|
511 | ## END
|
512 |
|
513 | #### "\e" as arg
|
514 | echo "${undef-\e}"
|
515 | ## STDOUT:
|
516 | \e
|
517 | ## END
|
518 | ## BUG zsh/mksh stdout-repr: '\x1b\n'
|
519 | ## BUG yash stdout: e
|
520 |
|