1 | ## compare_shells: bash-4.4 mksh
|
2 |
|
3 |
|
4 | # Extended assignment language, e.g. typeset, declare, arrays, etc.
|
5 | # Things that dash doesn't support.
|
6 |
|
7 | #### local -a
|
8 | # nixpkgs setup.sh uses this (issue #26)
|
9 | f() {
|
10 | local -a array=(x y z)
|
11 | argv.py "${array[@]}"
|
12 | }
|
13 | f
|
14 | ## stdout: ['x', 'y', 'z']
|
15 | ## N-I mksh stdout-json: ""
|
16 | ## N-I mksh status: 1
|
17 |
|
18 | #### declare -a
|
19 | # nixpkgs setup.sh uses this (issue #26)
|
20 | declare -a array=(x y z)
|
21 | argv.py "${array[@]}"
|
22 | ## stdout: ['x', 'y', 'z']
|
23 | ## N-I mksh stdout-json: ""
|
24 | ## N-I mksh status: 1
|
25 |
|
26 | #### indexed LHS with spaces (not allowed in OSH)
|
27 | a[1 * 1]=x a[ 1 + 2 ]=z
|
28 | echo status=$?
|
29 | argv.py "${a[@]}"
|
30 | ## STDOUT:
|
31 | status=0
|
32 | ['x', 'z']
|
33 | ## END
|
34 | ## N-I osh STDOUT:
|
35 | status=127
|
36 | []
|
37 | ## END
|
38 |
|
39 | #### declare -f exit code indicates function existence
|
40 | func2=x # var names are NOT found
|
41 | declare -f myfunc func2
|
42 | echo $?
|
43 |
|
44 | myfunc() { echo myfunc; }
|
45 | # This prints the source code.
|
46 | declare -f myfunc func2 > /dev/null
|
47 | echo $?
|
48 |
|
49 | func2() { echo func2; }
|
50 | declare -f myfunc func2 > /dev/null
|
51 | echo $?
|
52 | ## STDOUT:
|
53 | 1
|
54 | 1
|
55 | 0
|
56 | ## END
|
57 | ## N-I mksh STDOUT:
|
58 | 127
|
59 | 127
|
60 | 127
|
61 | ## END
|
62 |
|
63 | #### declare -F prints function names
|
64 | add () { expr 4 + 4; }
|
65 | div () { expr 6 / 2; }
|
66 | ek () { echo hello; }
|
67 | __ec () { echo hi; }
|
68 | _ab () { expr 10 % 3; }
|
69 |
|
70 | declare -F
|
71 | ## STDOUT:
|
72 | declare -f __ec
|
73 | declare -f _ab
|
74 | declare -f add
|
75 | declare -f div
|
76 | declare -f ek
|
77 | ## END
|
78 | ## N-I mksh stdout-json: ""
|
79 | ## N-I mksh status: 127
|
80 |
|
81 | #### declare -p var (exit status)
|
82 | var1() { echo func; } # function names are NOT found.
|
83 | declare -p var1 var2 >/dev/null
|
84 | echo $?
|
85 |
|
86 | var1=x
|
87 | declare -p var1 var2 >/dev/null
|
88 | echo $?
|
89 |
|
90 | var2=y
|
91 | declare -p var1 var2 >/dev/null
|
92 | echo $?
|
93 | ## STDOUT:
|
94 | 1
|
95 | 1
|
96 | 0
|
97 | ## N-I mksh STDOUT:
|
98 | 127
|
99 | 127
|
100 | 127
|
101 | ## END
|
102 |
|
103 | #### declare
|
104 | test_var1=111
|
105 | readonly test_var2=222
|
106 | export test_var3=333
|
107 | declare -n test_var4=test_var1
|
108 | f1() {
|
109 | local test_var5=555
|
110 | {
|
111 | echo '[declare]'
|
112 | declare
|
113 | echo '[readonly]'
|
114 | readonly
|
115 | echo '[export]'
|
116 | export
|
117 | echo '[local]'
|
118 | local
|
119 | } | grep -E '^\[|^\b.*test_var.\b'
|
120 | }
|
121 | f1
|
122 | ## STDOUT:
|
123 | [declare]
|
124 | test_var1=111
|
125 | test_var2=222
|
126 | test_var3=333
|
127 | test_var4=test_var1
|
128 | test_var5=555
|
129 | [readonly]
|
130 | declare -r test_var2=222
|
131 | [export]
|
132 | declare -x test_var3=333
|
133 | [local]
|
134 | test_var5=555
|
135 | ## END
|
136 | ## OK bash STDOUT:
|
137 | [declare]
|
138 | test_var1=111
|
139 | test_var2=222
|
140 | test_var3=333
|
141 | test_var4=test_var1
|
142 | test_var5=555
|
143 | [readonly]
|
144 | declare -r test_var2="222"
|
145 | [export]
|
146 | declare -x test_var3="333"
|
147 | [local]
|
148 | test_var5=555
|
149 | ## END
|
150 | ## N-I mksh STDOUT:
|
151 | [declare]
|
152 | [readonly]
|
153 | test_var2
|
154 | [export]
|
155 | test_var3
|
156 | [local]
|
157 | typeset test_var1
|
158 | typeset -r test_var2
|
159 | typeset -x test_var3
|
160 | typeset test_var5
|
161 | ## END
|
162 |
|
163 | #### declare -p
|
164 | # BUG: bash doesn't output flags with "local -p", which seems to contradict
|
165 | # with manual.
|
166 | test_var1=111
|
167 | readonly test_var2=222
|
168 | export test_var3=333
|
169 | declare -n test_var4=test_var1
|
170 | f1() {
|
171 | local test_var5=555
|
172 | {
|
173 | echo '[declare]'
|
174 | declare -p
|
175 | echo '[readonly]'
|
176 | readonly -p
|
177 | echo '[export]'
|
178 | export -p
|
179 | echo '[local]'
|
180 | local -p
|
181 | } | grep -E '^\[|^\b.*test_var.\b'
|
182 | }
|
183 | f1
|
184 | ## STDOUT:
|
185 | [declare]
|
186 | declare -- test_var1=111
|
187 | declare -r test_var2=222
|
188 | declare -x test_var3=333
|
189 | declare -n test_var4=test_var1
|
190 | declare -- test_var5=555
|
191 | [readonly]
|
192 | declare -r test_var2=222
|
193 | [export]
|
194 | declare -x test_var3=333
|
195 | [local]
|
196 | declare -- test_var5=555
|
197 | ## END
|
198 | ## BUG bash STDOUT:
|
199 | [declare]
|
200 | declare -- test_var1="111"
|
201 | declare -r test_var2="222"
|
202 | declare -x test_var3="333"
|
203 | declare -n test_var4="test_var1"
|
204 | declare -- test_var5="555"
|
205 | [readonly]
|
206 | declare -r test_var2="222"
|
207 | [export]
|
208 | declare -x test_var3="333"
|
209 | [local]
|
210 | test_var5=555
|
211 | ## END
|
212 | ## N-I mksh STDOUT:
|
213 | [declare]
|
214 | [readonly]
|
215 | readonly test_var2=222
|
216 | [export]
|
217 | export test_var3=333
|
218 | [local]
|
219 | typeset test_var1=111
|
220 | typeset -r test_var2=222
|
221 | typeset -x test_var3=333
|
222 | typeset test_var5=555
|
223 | ## END
|
224 |
|
225 | #### declare -p doesn't print binary data, but can be loaded into bash
|
226 |
|
227 | # bash prints binary data!
|
228 | case $SH in bash*|mksh) exit ;; esac
|
229 |
|
230 | unquoted='foo'
|
231 | sq='foo bar'
|
232 | bash1=$'\x1f' # ASCII control char
|
233 | bash2=$'\xfe\xff' # Invalid UTF-8
|
234 |
|
235 | s1=$unquoted
|
236 | s2=$sq
|
237 | s3=$bash1
|
238 | s4=$bash2
|
239 |
|
240 | declare -a a=("$unquoted" "$sq" "$bash1" "$bash2")
|
241 | declare -A A=(["$unquoted"]="$sq" ["$bash1"]="$bash2")
|
242 |
|
243 | #echo lengths ${#s1} ${#s2} ${#s3} ${#s4} ${#a[@]} ${#A[@]}
|
244 |
|
245 | declare -p s1 s2 s3 s4 a A | tee tmp.bash
|
246 |
|
247 | echo ---
|
248 |
|
249 | bash -c 'source tmp.bash; echo "$s1 $s2"; echo -n "$s3" "$s4" | od -A n -t x1'
|
250 | echo bash=$?
|
251 |
|
252 | ## STDOUT:
|
253 | declare -- s1=foo
|
254 | declare -- s2='foo bar'
|
255 | declare -- s3=$'\u001f'
|
256 | declare -- s4=$'\xfe\xff'
|
257 | declare -a a=(foo 'foo bar' $'\u001f' $'\xfe\xff')
|
258 | declare -A A=([$'\u001f']=$'\xfe\xff' ['foo']='foo bar')
|
259 | ---
|
260 | foo foo bar
|
261 | 1f 20 fe ff
|
262 | bash=0
|
263 | ## END
|
264 |
|
265 | ## N-I bash/mksh STDOUT:
|
266 | ## END
|
267 |
|
268 |
|
269 |
|
270 | #### declare -p var
|
271 | # BUG? bash doesn't output anything for 'local/readonly -p var', which seems to
|
272 | # contradict with manual. Besides, 'export -p var' is not described in
|
273 | # manual
|
274 | test_var1=111
|
275 | readonly test_var2=222
|
276 | export test_var3=333
|
277 | declare -n test_var4=test_var1
|
278 | f1() {
|
279 | local test_var5=555
|
280 | {
|
281 | echo '[declare]'
|
282 | declare -p test_var{0..5}
|
283 | echo '[readonly]'
|
284 | readonly -p test_var{0..5}
|
285 | echo '[export]'
|
286 | export -p test_var{0..5}
|
287 | echo '[local]'
|
288 | local -p test_var{0..5}
|
289 | } | grep -E '^\[|^\b.*test_var.\b'
|
290 | }
|
291 | f1
|
292 | ## STDOUT:
|
293 | [declare]
|
294 | declare -- test_var1=111
|
295 | declare -r test_var2=222
|
296 | declare -x test_var3=333
|
297 | declare -n test_var4=test_var1
|
298 | declare -- test_var5=555
|
299 | [readonly]
|
300 | declare -r test_var2=222
|
301 | [export]
|
302 | declare -x test_var3=333
|
303 | [local]
|
304 | declare -- test_var5=555
|
305 | ## END
|
306 | ## BUG bash STDOUT:
|
307 | [declare]
|
308 | declare -- test_var1="111"
|
309 | declare -r test_var2="222"
|
310 | declare -x test_var3="333"
|
311 | declare -n test_var4="test_var1"
|
312 | declare -- test_var5="555"
|
313 | [readonly]
|
314 | [export]
|
315 | [local]
|
316 | ## END
|
317 | ## N-I mksh STDOUT:
|
318 | [declare]
|
319 | [readonly]
|
320 | ## END
|
321 |
|
322 | #### declare -p arr
|
323 | test_arr1=()
|
324 | declare -a test_arr2=()
|
325 | declare -A test_arr3=()
|
326 | test_arr4=(1 2 3)
|
327 | declare -a test_arr5=(1 2 3)
|
328 | declare -A test_arr6=(['a']=1 ['b']=2 ['c']=3)
|
329 | test_arr7=()
|
330 | test_arr7[3]=foo
|
331 | declare -p test_arr{1..7}
|
332 | ## STDOUT:
|
333 | declare -a test_arr1=()
|
334 | declare -a test_arr2=()
|
335 | declare -A test_arr3=()
|
336 | declare -a test_arr4=(1 2 3)
|
337 | declare -a test_arr5=(1 2 3)
|
338 | declare -A test_arr6=(['a']=1 ['b']=2 ['c']=3)
|
339 | declare -a test_arr7=(); test_arr7[3]=foo
|
340 | ## END
|
341 | ## OK bash STDOUT:
|
342 | declare -a test_arr1=()
|
343 | declare -a test_arr2=()
|
344 | declare -A test_arr3=()
|
345 | declare -a test_arr4=([0]="1" [1]="2" [2]="3")
|
346 | declare -a test_arr5=([0]="1" [1]="2" [2]="3")
|
347 | declare -A test_arr6=([a]="1" [b]="2" [c]="3" )
|
348 | declare -a test_arr7=([3]="foo")
|
349 | ## END
|
350 | ## N-I mksh stdout-json: ""
|
351 | ## N-I mksh status: 1
|
352 |
|
353 | #### declare -p foo=bar doesn't make sense
|
354 | case $SH in (mksh) exit 0; esac
|
355 |
|
356 | declare -p foo=bar
|
357 | echo status=$?
|
358 |
|
359 | a=b
|
360 | declare -p a foo=bar > tmp.txt
|
361 | echo status=$?
|
362 | sed 's/"//g' tmp.txt # don't care about quotes
|
363 | ## STDOUT:
|
364 | status=1
|
365 | status=1
|
366 | declare -- a=b
|
367 | ## END
|
368 | ## N-I mksh stdout-json: ""
|
369 |
|
370 | #### declare -pnrx
|
371 | test_var1=111
|
372 | readonly test_var2=222
|
373 | export test_var3=333
|
374 | declare -n test_var4=test_var1
|
375 | f1() {
|
376 | local test_var5=555
|
377 | {
|
378 | echo '[declare -pn]'
|
379 | declare -pn
|
380 | echo '[declare -pr]'
|
381 | declare -pr
|
382 | echo '[declare -px]'
|
383 | declare -px
|
384 | } | grep -E '^\[|^\b.*test_var.\b'
|
385 | }
|
386 | f1
|
387 | ## STDOUT:
|
388 | [declare -pn]
|
389 | declare -n test_var4=test_var1
|
390 | [declare -pr]
|
391 | declare -r test_var2=222
|
392 | [declare -px]
|
393 | declare -x test_var3=333
|
394 | ## END
|
395 | ## OK bash STDOUT:
|
396 | [declare -pn]
|
397 | declare -n test_var4="test_var1"
|
398 | [declare -pr]
|
399 | declare -r test_var2="222"
|
400 | [declare -px]
|
401 | declare -x test_var3="333"
|
402 | ## END
|
403 | ## N-I mksh STDOUT:
|
404 | [declare -pn]
|
405 | [declare -pr]
|
406 | [declare -px]
|
407 | ## END
|
408 |
|
409 | #### declare -paA
|
410 | declare -a test_var6=()
|
411 | declare -A test_var7=()
|
412 | f1() {
|
413 | {
|
414 | echo '[declare -pa]'
|
415 | declare -pa
|
416 | echo '[declare -pA]'
|
417 | declare -pA
|
418 | } | grep -E '^\[|^\b.*test_var.\b'
|
419 | }
|
420 | f1
|
421 | ## STDOUT:
|
422 | [declare -pa]
|
423 | declare -a test_var6=()
|
424 | [declare -pA]
|
425 | declare -A test_var7=()
|
426 | ## END
|
427 | ## OK bash STDOUT:
|
428 | [declare -pa]
|
429 | declare -a test_var6=()
|
430 | [declare -pA]
|
431 | declare -A test_var7=()
|
432 | ## END
|
433 | ## N-I mksh stdout-json: ""
|
434 | ## N-I mksh status: 1
|
435 |
|
436 | #### declare -pnrx var
|
437 | # Note: Bash ignores other flags (-nrx) when variable names are supplied while
|
438 | # OSH uses other flags to select variables. Bash's behavior is documented.
|
439 | test_var1=111
|
440 | readonly test_var2=222
|
441 | export test_var3=333
|
442 | declare -n test_var4=test_var1
|
443 | f1() {
|
444 | local test_var5=555
|
445 | {
|
446 | echo '[declare -pn]'
|
447 | declare -pn test_var{0..5}
|
448 | echo '[declare -pr]'
|
449 | declare -pr test_var{0..5}
|
450 | echo '[declare -px]'
|
451 | declare -px test_var{0..5}
|
452 | } | grep -E '^\[|^\b.*test_var.\b'
|
453 | }
|
454 | f1
|
455 | ## STDOUT:
|
456 | [declare -pn]
|
457 | declare -n test_var4=test_var1
|
458 | [declare -pr]
|
459 | declare -r test_var2=222
|
460 | [declare -px]
|
461 | declare -x test_var3=333
|
462 | ## END
|
463 | ## N-I bash STDOUT:
|
464 | [declare -pn]
|
465 | declare -- test_var1="111"
|
466 | declare -r test_var2="222"
|
467 | declare -x test_var3="333"
|
468 | declare -n test_var4="test_var1"
|
469 | declare -- test_var5="555"
|
470 | [declare -pr]
|
471 | declare -- test_var1="111"
|
472 | declare -r test_var2="222"
|
473 | declare -x test_var3="333"
|
474 | declare -n test_var4="test_var1"
|
475 | declare -- test_var5="555"
|
476 | [declare -px]
|
477 | declare -- test_var1="111"
|
478 | declare -r test_var2="222"
|
479 | declare -x test_var3="333"
|
480 | declare -n test_var4="test_var1"
|
481 | declare -- test_var5="555"
|
482 | ## END
|
483 | ## N-I mksh STDOUT:
|
484 | [declare -pn]
|
485 | [declare -pr]
|
486 | [declare -px]
|
487 | ## END
|
488 |
|
489 | #### declare -pg
|
490 | test_var1=global
|
491 | f1() {
|
492 | local test_var1=local
|
493 | {
|
494 | declare -pg
|
495 | } | grep -E '^\[|^\b[^"]*test_var.\b'
|
496 | }
|
497 | f1
|
498 | ## STDOUT:
|
499 | declare -- test_var1=global
|
500 | ## END
|
501 | ## N-I bash STDOUT:
|
502 | declare -- test_var1="local"
|
503 | ## END
|
504 | ## N-I mksh stdout-json: ""
|
505 | ## N-I mksh status: 1
|
506 |
|
507 | #### declare -pg var
|
508 | test_var1=global
|
509 | f1() {
|
510 | local test_var1=local
|
511 | {
|
512 | declare -pg test_var1
|
513 | } | grep -E '^\[|^\b.*test_var.\b'
|
514 | }
|
515 | f1
|
516 | ## STDOUT:
|
517 | declare -- test_var1=global
|
518 | ## END
|
519 | ## N-I bash STDOUT:
|
520 | declare -- test_var1="local"
|
521 | ## END
|
522 | ## N-I mksh stdout-json: ""
|
523 | ## N-I mksh status: 1
|
524 |
|
525 | #### ble.sh: eval -- "$(declare -p var arr)"
|
526 | # This illustrates an example usage of "eval & declare" for exporting
|
527 | # multiple variables from $().
|
528 | eval -- "$(
|
529 | printf '%s\n' a{1..10} | {
|
530 | sum=0 i=0 arr=()
|
531 | while read line; do
|
532 | ((sum+=${#line},i++))
|
533 | arr[$((i/3))]=$line
|
534 | done
|
535 | declare -p sum arr
|
536 | })"
|
537 | echo sum=$sum
|
538 | for ((i=0;i<${#arr[@]};i++)); do
|
539 | echo "arr[$i]=${arr[i]}"
|
540 | done
|
541 | ## STDOUT:
|
542 | sum=21
|
543 | arr[0]=a2
|
544 | arr[1]=a5
|
545 | arr[2]=a8
|
546 | arr[3]=a10
|
547 | ## END
|
548 | ## N-I mksh stdout-json: ""
|
549 | ## N-I mksh status: 1
|
550 |
|
551 | #### eval -- "$(declare -p arr)" (restore arrays w/ unset elements)
|
552 | arr=(1 2 3)
|
553 | eval -- "$(arr=(); arr[3]= arr[4]=foo; declare -p arr)"
|
554 | for i in {0..4}; do
|
555 | echo "arr[$i]: ${arr[$i]+set ... [}${arr[$i]-unset}${arr[$i]+]}"
|
556 | done
|
557 | ## STDOUT:
|
558 | arr[0]: unset
|
559 | arr[1]: unset
|
560 | arr[2]: unset
|
561 | arr[3]: set ... []
|
562 | arr[4]: set ... [foo]
|
563 | ## END
|
564 | ## N-I mksh stdout-json: ""
|
565 | ## N-I mksh status: 1
|
566 |
|
567 | #### typeset -f
|
568 | # mksh implement typeset but not declare
|
569 | typeset -f myfunc func2
|
570 | echo $?
|
571 |
|
572 | myfunc() { echo myfunc; }
|
573 | # This prints the source code.
|
574 | typeset -f myfunc func2 > /dev/null
|
575 | echo $?
|
576 |
|
577 | func2() { echo func2; }
|
578 | typeset -f myfunc func2 > /dev/null
|
579 | echo $?
|
580 | ## STDOUT:
|
581 | 1
|
582 | 1
|
583 | 0
|
584 | ## END
|
585 |
|
586 | #### typeset -p
|
587 | var1() { echo func; } # function names are NOT found.
|
588 | typeset -p var1 var2 >/dev/null
|
589 | echo $?
|
590 |
|
591 | var1=x
|
592 | typeset -p var1 var2 >/dev/null
|
593 | echo $?
|
594 |
|
595 | var2=y
|
596 | typeset -p var1 var2 >/dev/null
|
597 | echo $?
|
598 | ## STDOUT:
|
599 | 1
|
600 | 1
|
601 | 0
|
602 | ## BUG mksh STDOUT:
|
603 | # mksh doesn't respect exit codes
|
604 | 0
|
605 | 0
|
606 | 0
|
607 | ## END
|
608 |
|
609 | #### typeset -r makes a string readonly
|
610 | typeset -r s1='12'
|
611 | typeset -r s2='34'
|
612 |
|
613 | s1='c'
|
614 | echo status=$?
|
615 | s2='d'
|
616 | echo status=$?
|
617 |
|
618 | s1+='e'
|
619 | echo status=$?
|
620 | s2+='f'
|
621 | echo status=$?
|
622 |
|
623 | unset s1
|
624 | echo status=$?
|
625 | unset s2
|
626 | echo status=$?
|
627 |
|
628 | ## status: 1
|
629 | ## stdout-json: ""
|
630 | ## OK mksh status: 2
|
631 | ## OK bash status: 0
|
632 | ## OK bash STDOUT:
|
633 | status=1
|
634 | status=1
|
635 | status=1
|
636 | status=1
|
637 | status=1
|
638 | status=1
|
639 | ## END
|
640 |
|
641 | #### typeset -ar makes it readonly
|
642 | typeset -a -r array1=(1 2)
|
643 | typeset -ar array2=(3 4)
|
644 |
|
645 | array1=('c')
|
646 | echo status=$?
|
647 | array2=('d')
|
648 | echo status=$?
|
649 |
|
650 | array1+=('e')
|
651 | echo status=$?
|
652 | array2+=('f')
|
653 | echo status=$?
|
654 |
|
655 | unset array1
|
656 | echo status=$?
|
657 | unset array2
|
658 | echo status=$?
|
659 |
|
660 | ## status: 1
|
661 | ## stdout-json: ""
|
662 | ## OK bash status: 0
|
663 | ## OK bash STDOUT:
|
664 | status=1
|
665 | status=1
|
666 | status=1
|
667 | status=1
|
668 | status=1
|
669 | status=1
|
670 | ## END
|
671 | ## N-I mksh status: 1
|
672 | ## N-I mksh stdout-json: ""
|
673 |
|
674 | #### typeset -x makes it exported
|
675 | typeset -rx PYTHONPATH=lib/
|
676 | printenv.py PYTHONPATH
|
677 | ## STDOUT:
|
678 | lib/
|
679 | ## END
|
680 |
|
681 | #### Multiple assignments / array assignments on a line
|
682 | a=1 b[0+0]=2 c=3
|
683 | echo $a ${b[@]} $c
|
684 | ## stdout: 1 2 3
|
685 |
|
686 | #### Env bindings shouldn't contain array assignments
|
687 | a=1 b[0]=2 c=3 printenv.py a b c
|
688 | ## status: 2
|
689 | ## stdout-json: ""
|
690 | ## OK bash STDOUT:
|
691 | 1
|
692 | None
|
693 | 3
|
694 | ## END
|
695 | ## OK bash status: 0
|
696 | ## BUG mksh STDOUT:
|
697 | 1
|
698 | 2
|
699 | 3
|
700 | ## END
|
701 | ## BUG mksh status: 0
|
702 |
|
703 | #### syntax error in array assignment
|
704 | a=x b[0+]=y c=z
|
705 | echo $a $b $c
|
706 | ## status: 2
|
707 | ## stdout-json: ""
|
708 | ## BUG bash stdout: x
|
709 | ## BUG bash status: 0
|
710 | ## OK mksh stdout-json: ""
|
711 | ## OK mksh status: 1
|
712 |
|
713 | #### declare -g (bash-specific; bash-completion uses it)
|
714 | f() {
|
715 | declare -g G=42
|
716 | declare L=99
|
717 |
|
718 | declare -Ag dict
|
719 | dict["foo"]=bar
|
720 |
|
721 | declare -A localdict
|
722 | localdict["spam"]=Eggs
|
723 |
|
724 | # For bash-completion
|
725 | eval 'declare -Ag ev'
|
726 | ev["ev1"]=ev2
|
727 | }
|
728 | f
|
729 | argv.py "$G" "$L"
|
730 | argv.py "${dict["foo"]}" "${localdict["spam"]}"
|
731 | argv.py "${ev["ev1"]}"
|
732 | ## STDOUT:
|
733 | ['42', '']
|
734 | ['bar', '']
|
735 | ['ev2']
|
736 | ## END
|
737 | ## N-I mksh STDOUT:
|
738 | ['', '']
|
739 | ## END
|
740 | ## N-I mksh status: 1
|
741 |
|
742 | #### myvar=typeset (another form of dynamic assignment)
|
743 | myvar=typeset
|
744 | x='a b'
|
745 | $myvar x=$x
|
746 | echo $x
|
747 | ## STDOUT:
|
748 | a
|
749 | ## END
|
750 | ## OK osh STDOUT:
|
751 | a b
|
752 | ## END
|
753 |
|
754 | #### dynamic array parsing is not allowed
|
755 | code='x=(1 2 3)'
|
756 | typeset -a "$code" # note: -a flag is required
|
757 | echo status=$?
|
758 | argv.py "$x"
|
759 | ## STDOUT:
|
760 | status=2
|
761 | ['']
|
762 | ## END
|
763 | ## OK mksh STDOUT:
|
764 | status=0
|
765 | ['(1 2 3)']
|
766 | ## END
|
767 | # bash allows it
|
768 | ## OK bash STDOUT:
|
769 | status=0
|
770 | ['1']
|
771 | ## END
|
772 |
|
773 | #### dynamic flag in array in assign builtin
|
774 | typeset b
|
775 | b=(unused1 unused2) # this works in mksh
|
776 |
|
777 | a=(x 'foo=F' 'bar=B')
|
778 | typeset -"${a[@]}"
|
779 | echo foo=$foo
|
780 | echo bar=$bar
|
781 | printenv.py foo
|
782 | printenv.py bar
|
783 |
|
784 | # syntax error in mksh! But works in bash and zsh.
|
785 | #typeset -"${a[@]}" b=(spam eggs)
|
786 | #echo "length of b = ${#b[@]}"
|
787 | #echo "b[0]=${b[0]}"
|
788 | #echo "b[1]=${b[1]}"
|
789 |
|
790 | ## STDOUT:
|
791 | foo=F
|
792 | bar=B
|
793 | F
|
794 | B
|
795 | ## END
|
796 |
|
797 | #### typeset +x
|
798 | export e=E
|
799 | printenv.py e
|
800 | typeset +x e=E2
|
801 | printenv.py e # no longer exported
|
802 | ## STDOUT:
|
803 | E
|
804 | None
|
805 | ## END
|
806 |
|
807 | #### typeset +r removes read-only attribute (TODO: documented in bash to do nothing)
|
808 | readonly r=r1
|
809 | echo r=$r
|
810 |
|
811 | # clear the readonly flag. Why is this accepted in bash, but doesn't do
|
812 | # anything?
|
813 | typeset +r r=r2
|
814 | echo r=$r
|
815 |
|
816 | r=r3
|
817 | echo r=$r
|
818 |
|
819 | ## status: 0
|
820 | ## STDOUT:
|
821 | r=r1
|
822 | r=r2
|
823 | r=r3
|
824 | ## END
|
825 |
|
826 | # mksh doesn't allow you to unset
|
827 | ## OK mksh status: 2
|
828 | ## OK mksh STDOUT:
|
829 | r=r1
|
830 | ## END
|
831 |
|
832 | # bash doesn't allow you to unset
|
833 | ## OK bash status: 0
|
834 | ## OK bash STDOUT:
|
835 | r=r1
|
836 | r=r1
|
837 | r=r1
|
838 | ## END
|
839 |
|
840 |
|
841 | #### function name with /
|
842 | ble/foo() { echo hi; }
|
843 | declare -F ble/foo
|
844 | echo status=$?
|
845 | ## STDOUT:
|
846 | ble/foo
|
847 | status=0
|
848 | ## END
|
849 | ## N-I mksh stdout: status=127
|
850 | ## N-I zsh stdout-json: ""
|
851 | ## N-I zsh status: 1
|
852 | ## N-I ash stdout-json: ""
|
853 | ## N-I ash status: 2
|
854 |
|
855 | #### invalid var name
|
856 | typeset foo/bar
|
857 | ## status: 1
|
858 |
|
859 | #### unset and shell funcs
|
860 | foo() {
|
861 | echo bar
|
862 | }
|
863 |
|
864 | foo
|
865 |
|
866 | declare -F
|
867 | unset foo
|
868 | declare -F
|
869 |
|
870 | foo
|
871 |
|
872 | ## status: 127
|
873 | ## STDOUT:
|
874 | bar
|
875 | declare -f foo
|
876 | ## END
|
877 | ## N-I mksh status: 0
|
878 | ## N-I mksh STDOUT:
|
879 | bar
|
880 | bar
|
881 | ## END
|