1 ## compare_shells: bash-4.4
2 ## oils_failures_allowed: 4
3
4
5 # NOTE:
6 # -declare -A is required.
7 #
8 # Simply doing:
9 # a=([aa]=b [foo]=bar ['a+1']=c)
10 # gets utterly bizarre behavior.
11 #
12 # Associtative Arrays are COMPLETELY bash-specific. mksh doesn't even come
13 # close. So I will probably not implement them, or implement something
14 # slightly different, because the semantics are just weird.
15
16 # http://www.gnu.org/software/bash/manual/html_node/Arrays.html
17 # TODO: Need a SETUP section.
18
19 #### Literal syntax ([x]=y)
20 declare -A a
21 a=([aa]=b [foo]=bar ['a+1']=c)
22 echo ${a["aa"]}
23 echo ${a["foo"]}
24 echo ${a["a+1"]}
25 ## STDOUT:
26 b
27 bar
28 c
29 ## END
30
31 #### set associative array to indexed array literal (very surprising bash behavior)
32 declare -A assoc=([k1]=foo [k2]='spam eggs')
33 for v in "${assoc[@]}"; do echo $v; done | sort
34 for v in "${!assoc[@]}"; do echo $v; done | sort
35
36 # disallow this in OSH? Changing type?
37
38 assoc=(foo 'spam eggs')
39 argv.py "${assoc[@]}"
40 argv.py "${!assoc[@]}"
41
42 ## STDOUT:
43 foo
44 spam eggs
45 k1
46 k2
47 ['foo', 'spam eggs']
48 ['0', '1']
49 ## END
50 ## BUG bash STDOUT:
51 foo
52 spam eggs
53 k1
54 k2
55 []
56 []
57 ## END
58
59 #### Can't initialize assoc array with indexed array
60 declare -A A=(1 2 3)
61 echo status=$?
62 ## STDOUT:
63 status=2
64 ## END
65
66 # bash prints warnings to stderr but gives no indication of the problem
67 ## BUG bash STDOUT:
68 status=0
69 ## END
70
71
72 #### Initializing indexed array with assoc array
73 declare -a a=([xx]=1 [yy]=2 [zz]=3)
74 echo status=$?
75 argv.py "${a[@]}"
76 ## STDOUT:
77 status=2
78 []
79 ## END
80 ## BUG bash STDOUT:
81 status=0
82 ['3']
83 ## END
84
85 #### create empty assoc array, put, then get
86 declare -A A # still undefined
87 argv.py "${A[@]}"
88 argv.py "${!A[@]}"
89 A['foo']=bar
90 echo ${A['foo']}
91 ## STDOUT:
92 []
93 []
94 bar
95 ## END
96
97 #### Empty value (doesn't use EmptyWord?)
98 declare -A A=(["k"]= )
99 argv.py "${A["k"]}"
100 ## STDOUT:
101 ['']
102 ## END
103
104 #### retrieve keys with !
105 declare -A a
106 var='x'
107 a["$var"]=b
108 a['foo']=bar
109 a['a+1']=c
110 for key in "${!a[@]}"; do
111 echo $key
112 done | sort
113 ## STDOUT:
114 a+1
115 foo
116 x
117 ## END
118
119 #### retrieve values with ${A[@]}
120 declare -A A
121 var='x'
122 A["$var"]=b
123 A['foo']=bar
124 A['a+1']=c
125 for val in "${A[@]}"; do
126 echo $val
127 done | sort
128 ## STDOUT:
129 b
130 bar
131 c
132 ## END
133
134 #### coerce to string with ${A[*]}, etc.
135 declare -A A
136 A['X X']=xx
137 A['Y Y']=yy
138 argv.py "${A[*]}"
139 argv.py "${!A[*]}"
140
141 argv.py ${A[@]}
142 argv.py ${!A[@]}
143 ## STDOUT:
144 ['xx yy']
145 ['X X Y Y']
146 ['xx', 'yy']
147 ['X', 'X', 'Y', 'Y']
148 ## END
149
150 #### ${A[@]/b/B}
151 # but ${!A[@]/b/B} doesn't work
152 declare -A A
153 A['aa']=bbb
154 A['bb']=ccc
155 A['cc']=ddd
156 for val in "${A[@]//b/B}"; do
157 echo $val
158 done | sort
159 ## STDOUT:
160 BBB
161 ccc
162 ddd
163 ## END
164
165 #### ${A[@]#prefix}
166 declare -A A
167 A['aa']=one
168 A['bb']=two
169 A['cc']=three
170 for val in "${A[@]#t}"; do
171 echo $val
172 done | sort
173 ## STDOUT:
174 hree
175 one
176 wo
177 ## END
178
179 #### ${assoc} is like ${assoc[0]}
180 declare -A a
181
182 a=([aa]=b [foo]=bar ['a+1']=c)
183 echo a="${a}"
184
185 a=([0]=zzz)
186 echo a="${a}"
187
188 a=(['0']=yyy)
189 echo a="${a}"
190
191 ## STDOUT:
192 a=
193 a=zzz
194 a=yyy
195 ## END
196
197 #### length ${#a[@]}
198 declare -A a
199 a["x"]=1
200 a["y"]=2
201 a["z"]=3
202 echo "${#a[@]}"
203 ## stdout: 3
204
205 #### lookup with ${a[0]} -- "0" is a string
206 declare -A a
207 a["0"]=a
208 a["1"]=b
209 a["2"]=c
210 echo 0 "${a[0]}" 1 "${a[1]}" 2 "${a[2]}"
211 ## STDOUT:
212 0 a 1 b 2 c
213 ## END
214
215 #### lookup with double quoted strings "mykey"
216 declare -A a
217 a["aa"]=b
218 a["foo"]=bar
219 a['a+1']=c
220 echo "${a["aa"]}" "${a["foo"]}" "${a["a+1"]}"
221 ## STDOUT:
222 b bar c
223 ## END
224
225 #### lookup with single quoted string
226 declare -A a
227 a["aa"]=b
228 a["foo"]=bar
229 a['a+1']=c
230 echo "${a['a+1']}"
231 ## stdout: c
232
233 #### lookup with unquoted $key and quoted "$i$i"
234 declare -A A
235 A["aa"]=b
236 A["foo"]=bar
237
238 key=foo
239 echo ${A[$key]}
240 i=a
241 echo ${A["$i$i"]} # note: ${A[$i$i]} doesn't work in OSH
242 ## STDOUT:
243 bar
244 b
245 ## END
246
247 #### lookup by unquoted string doesn't work in OSH because it's a variable
248 declare -A a
249 a["aa"]=b
250 a["foo"]=bar
251 a['a+1']=c
252 echo "${a[a+1]}"
253 ## stdout-json: ""
254 ## status: 1
255 ## BUG bash stdout: c
256 ## BUG bash status: 0
257
258 #### bash bug: "i+1" and i+1 are the same key
259
260 i=1
261 array=(5 6 7)
262 echo array[i]="${array[i]}"
263 echo array[i+1]="${array[i+1]}"
264
265 # arithmetic does NOT work here in bash. These are unquoted strings!
266 declare -A assoc
267 assoc[i]=$i
268 assoc[i+1]=$i+1
269
270 assoc["i"]=string
271 assoc["i+1"]=string+1
272
273 echo assoc[i]="${assoc[i]}"
274 echo assoc[i+1]="${assoc[i+1]}"
275
276 echo assoc[i]="${assoc["i"]}"
277 echo assoc[i+1]="${assoc["i+1"]}"
278
279 ## status: 1
280 ## STDOUT:
281 array[i]=6
282 array[i+1]=7
283 ## END
284 ## BUG bash status: 0
285 ## BUG bash STDOUT:
286 array[i]=6
287 array[i+1]=7
288 assoc[i]=string
289 assoc[i+1]=string+1
290 assoc[i]=string
291 assoc[i+1]=string+1
292 ## END
293
294 #### Array stored in associative array gets converted to string (without strict_array)
295
296 array=('1 2' 3)
297 declare -A d
298 d['key']="${array[@]}"
299 argv.py "${d['key']}"
300 ## stdout: ['1 2 3']
301
302 #### Indexed array as key of associative array coerces to string (without shopt -s strict_array)
303
304 declare -a array=(1 2 3)
305 declare -A assoc
306 assoc[42]=43
307 assoc["${array[@]}"]=foo
308
309 echo "${assoc["${array[@]}"]}"
310 for entry in "${!assoc[@]}"; do
311 echo $entry
312 done | sort
313
314 ## STDOUT:
315 foo
316 1 2 3
317 42
318 ## END
319
320 #### Append to associative array value A['x']+='suffix'
321 declare -A A
322 A['x']='foo'
323 A['x']+='bar'
324 A['x']+='bar'
325 argv.py "${A["x"]}"
326 ## STDOUT:
327 ['foobarbar']
328 ## END
329
330 #### Slice of associative array doesn't make sense in bash
331 declare -A a
332 a[xx]=1
333 a[yy]=2
334 a[zz]=3
335 a[aa]=4
336 a[bb]=5
337 #argv.py ${a["xx"]}
338 argv.py ${a[@]: 0: 3}
339 argv.py ${a[@]: 1: 3}
340 argv.py ${a[@]: 2: 3}
341 argv.py ${a[@]: 3: 3}
342 argv.py ${a[@]: 4: 3}
343 argv.py ${a[@]: 5: 3}
344 ## stdout-json: ""
345 ## status: 1
346 ## BUG bash STDOUT:
347 ['2', '1', '5']
348 ['2', '1', '5']
349 ['1', '5', '4']
350 ['5', '4', '3']
351 ['4', '3']
352 ['3']
353 ## END
354 ## BUG bash status: 0
355
356 #### bash variable can have an associative array part and a string part
357
358 # and $assoc is equivalent to ${assoc[0]}, just like regular arrays
359 declare -A assoc
360 assoc[1]=1
361 assoc[2]=2
362 echo ${assoc[1]} ${assoc[2]} ${assoc}
363 assoc[0]=zero
364 echo ${assoc[1]} ${assoc[2]} ${assoc}
365 assoc=string
366 echo ${assoc[1]} ${assoc[2]} ${assoc}
367 ## STDOUT:
368 1 2
369 1 2 zero
370 1 2 string
371 ## END
372 ## N-I osh status: 1
373 ## N-I osh STDOUT:
374 1 2
375 1 2 zero
376 ## END
377
378 #### Associative array expressions inside (( )) with keys that look like numbers
379 declare -A assoc
380 assoc[0]=42
381 (( var = ${assoc[0]} ))
382 echo $var
383 (( var = assoc[0] ))
384 echo $var
385 ## STDOUT:
386 42
387 42
388 ## END
389
390 #### (( A[5] += 42 ))
391 declare -A A
392 (( A[5] = 10 ))
393 (( A[5] += 6 ))
394 echo ${A[5]}
395 ## STDOUT:
396 16
397 ## END
398
399 #### (( A[5] += 42 )) with empty cell
400 shopt -u strict_arith # default zero cell
401 declare -A A
402 (( A[5] += 6 ))
403 echo ${A[5]}
404 ## STDOUT:
405 6
406 ## END
407
408 #### setting key to itself (from bash-bug mailing list)
409 declare -A foo
410 foo=(["key"]="value1")
411 echo ${foo["key"]}
412 foo=(["key"]="${foo["key"]} value2")
413 echo ${foo["key"]}
414 ## STDOUT:
415 value1
416 value1 value2
417 ## END
418 ## BUG bash STDOUT:
419 value1
420 value2
421 ## END
422
423 #### readonly associative array can't be modified
424 declare -Ar A
425 A['x']=1
426 echo status=$?
427 ## OK osh status: 1
428 ## OK osh stdout-json: ""
429 ## STDOUT:
430 status=1
431 ## END
432
433 #### associative array and brace expansion
434 declare -A A=([k1]=v [k2]=-{a,b}-)
435 echo ${A["k1"]}
436 echo ${A["k2"]}
437 ## STDOUT:
438 v
439 -{a,b}-
440 ## END
441
442 #### bash mangles array #1
443 a=([k1]=v1 [k2]=v2)
444 echo ${a["k1"]}
445 echo ${a["k2"]}
446 ## STDOUT:
447 v1
448 v2
449 ## END
450 ## BUG bash STDOUT:
451 v2
452 v2
453 ## END
454
455 #### bash mangles array and brace #2
456 a=([k2]=-{a,b}-)
457 echo ${a["k2"]}
458 ## STDOUT:
459 -{a,b}-
460 ## END
461 ## BUG bash STDOUT:
462 [k2]=-a-
463 ## END
464
465 #### declare -A A=() allowed
466 set -o nounset
467 shopt -s strict_arith || true
468
469 declare -A ASSOC=()
470 echo len=${#ASSOC[@]}
471
472 # Check that it really can be used like an associative array
473 ASSOC['k']='32'
474 echo len=${#ASSOC[@]}
475
476 # bash allows a variable to be an associative array AND unset, while OSH
477 # doesn't
478 set +o nounset
479 declare -A u
480 echo unset len=${#u[@]}
481 ## STDOUT:
482 len=0
483 len=1
484 unset len=0
485 ## END
486
487 #### unset -v and assoc array
488 shopt -s eval_unsafe_arith || true
489
490 show-len() {
491 echo len=${#assoc[@]}
492 }
493
494 declare -A assoc=(['K']=val)
495 show-len
496
497 unset -v 'assoc["K"]'
498 show-len
499
500 declare -A assoc=(['K']=val)
501 show-len
502 key=K
503 unset -v 'assoc[$key]'
504 show-len
505
506 declare -A assoc=(['K']=val)
507 show-len
508 unset -v 'assoc[$(echo K)]'
509 show-len
510
511 # ${prefix} doesn't work here, even though it does in arithmetic
512 #declare -A assoc=(['K']=val)
513 #show-len
514 #prefix=as
515 #unset -v '${prefix}soc[$key]'
516 #show-len
517
518 ## STDOUT:
519 len=1
520 len=0
521 len=1
522 len=0
523 len=1
524 len=0
525 ## END
526
527 #### nameref and assoc array
528 show-values() {
529 echo values: ${A[@]}
530 }
531
532 declare -A A=(['K']=val)
533 show-values
534
535 declare -n ref='A["K"]'
536 echo before $ref
537 ref='val2'
538 echo after $ref
539 show-values
540
541 echo ---
542
543 key=K
544 declare -n ref='A[$key]'
545 echo before $ref
546 ref='val3'
547 echo after $ref
548 show-values
549
550 ## STDOUT:
551 values: val
552 before val
553 after val2
554 values: val2
555 ---
556 before val2
557 after val3
558 values: val3
559 ## END
560
561 #### ${!ref} and assoc array
562
563 show-values() {
564 echo values: ${A[@]}
565 }
566
567 declare -A A=(['K']=val)
568 show-values
569
570 declare ref='A["K"]'
571 echo ref ${!ref}
572
573 key=K
574 declare ref='A[$key]'
575 echo ref ${!ref}
576
577 ## STDOUT:
578 values: val
579 ref val
580 ref val
581 ## END
582
583 #### printf -v and assoc array
584
585 show-values() {
586 echo values: ${assoc[@]}
587 }
588
589 declare -A assoc=(['K']=val)
590 show-values
591
592 printf -v 'assoc["K"]' '/%s/' val2
593 show-values
594
595 key=K
596 printf -v 'assoc[$key]' '/%s/' val3
597 show-values
598
599 # Somehow bash doesn't allow this
600 #prefix=as
601 #printf -v '${prefix}soc[$key]' '/%s/' val4
602 #show-values
603
604 ## STDOUT:
605 values: val
606 values: /val2/
607 values: /val3/
608 ## END
609
610 #### bash bug: (( A["$key"] = 1 )) doesn't work
611 key='\'
612 declare -A A
613 #A["$key"]=1
614
615 # Works in both
616 #A["$key"]=42
617
618 # Works in bash only
619 #(( A[\$key] = 42 ))
620
621 (( A["$key"] = 42 ))
622
623 argv.py "${!A[@]}"
624 argv.py "${A[@]}"
625 ## STDOUT:
626 ['\\']
627 ['42']
628 ## END
629 ## BUG bash STDOUT:
630 []
631 []
632 ## END
633
634
635 #### Implicit increment of keys
636 declare -a arr=( [30]=a b [40]=x y)
637 argv.py "${!arr[@]}"
638 argv.py "${arr[@]}"
639
640 # osh says "expected associative array pair"
641
642 ## STDOUT:
643 ['30', '31', '40', '41']
644 ['a', 'b', 'x', 'y']
645 ## END
646
647 #### test -v assoc[key]
648
649 typeset -A assoc
650 assoc=([empty]='' [k]=v)
651
652 echo 'no quotes'
653
654 test -v assoc[empty]
655 echo empty=$?
656
657 test -v assoc[k]
658 echo k=$?
659
660 test -v assoc[nonexistent]
661 echo nonexistent=$?
662
663 echo
664
665 # Now with quotes
666 echo 'quotes'
667
668 test -v assoc["empty"]
669 echo empty=$?
670
671 test -v assoc['k']
672 echo k=$?
673
674 test -v assoc['nonexistent']
675 echo nonexistent=$?
676
677 ## STDOUT:
678 no quotes
679 empty=0
680 k=0
681 nonexistent=1
682
683 quotes
684 empty=0
685 k=0
686 nonexistent=1
687 ## END
688
689 #### test -v with dynamic parsing
690
691 typeset -A assoc
692 assoc=([empty]='' [k]=v)
693
694 key=empty
695 test -v 'assoc[$key]'
696 echo empty=$?
697
698 key=k
699 test -v 'assoc[$key]'
700 echo k=$?
701
702 key=nonexistent
703 test -v 'assoc[$key]'
704 echo nonexistent=$?
705
706 ## STDOUT:
707 empty=0
708 k=0
709 nonexistent=1
710 ## END
711
712 #### [[ -v assoc[key] ]]
713
714 typeset -A assoc
715 assoc=([empty]='' [k]=v)
716
717 echo 'no quotes'
718
719 [[ -v assoc[empty] ]]
720 echo empty=$?
721
722 [[ -v assoc[k] ]]
723 echo k=$?
724
725 [[ -v assoc[nonexistent] ]]
726 echo nonexistent=$?
727
728 echo
729
730 # Now with quotes
731 echo 'quotes'
732
733 [[ -v assoc["empty"] ]]
734 echo empty=$?
735
736 [[ -v assoc['k'] ]]
737 echo k=$?
738
739 [[ -v assoc['nonexistent'] ]]
740 echo nonexistent=$?
741
742 echo
743
744 echo 'vars'
745
746 key=empty
747 [[ -v assoc[$key] ]]
748 echo empty=$?
749
750 key=k
751 [[ -v assoc[$key] ]]
752 echo k=$?
753
754 key=nonexistent
755 [[ -v assoc[$key] ]]
756 echo nonexistent=$?
757
758 ## STDOUT:
759 no quotes
760 empty=0
761 k=0
762 nonexistent=1
763
764 quotes
765 empty=0
766 k=0
767 nonexistent=1
768
769 vars
770 empty=0
771 k=0
772 nonexistent=1
773 ## END
774
775 ## N-I mksh status: 1
776 ## N-I mksh STDOUT:
777 ## END
778
779 #### [[ -v assoc[key] ]] syntax errors
780
781 typeset -A assoc
782 assoc=([empty]='' [k]=v)
783
784 [[ -v assoc[empty] ]]
785 echo empty=$?
786
787 [[ -v assoc[k] ]]
788 echo k=$?
789
790 [[ -v assoc[k]z ]]
791 echo typo=$?
792
793 ## STDOUT:
794 empty=0
795 k=0
796 typo=1
797 ## END
798
799
800 #### BashAssoc a+=()
801
802 declare -A a=([apple]=red [orange]=orange)
803 a+=([lemon]=yellow [banana]=yellow)
804 echo "apple is ${a['apple']}"
805 echo "orange is ${a['orange']}"
806 echo "lemon is ${a['lemon']}"
807 echo "banana is ${a['banana']}"
808
809 ## STDOUT:
810 apple is red
811 orange is orange
812 lemon is yellow
813 banana is yellow
814 ## END
815
816
817 #### BashAssoc ${a[@]@Q}
818
819 declare -A a=()
820 a['symbol1']=\'\'
821 a['symbol2']='"'
822 a['symbol3']='()<>&|'
823 a['symbol4']='[]*?'
824 echo "[${a[@]@Q}]"
825 echo "[${a[*]@Q}]"
826
827 ## STDOUT:
828 [$'\'\'' '"' '()<>&|' '[]*?']
829 [$'\'\'' '"' '()<>&|' '[]*?']
830 ## END
831
832 ## OK bash STDOUT:
833 ['[]*?' ''\'''\''' '"' '()<>&|']
834 ['[]*?' ''\'''\''' '"' '()<>&|']
835 ## END