1 ## oils_failures_allowed: 7
2 ## compare_shells: bash mksh
3
4 #### pass array by reference
5 show_value() {
6 local -n array_name=$1
7 local idx=$2
8 echo "${array_name[$idx]}"
9 }
10 shadock=(ga bu zo meu)
11 show_value shadock 2
12 ## stdout: zo
13
14 #### mutate array by reference
15 set1() {
16 local -n array_name=$1
17 local val=$2
18 array_name[1]=$val
19 }
20 shadock=(a b c d)
21 set1 shadock ZZZ
22 echo ${shadock[@]}
23 ## STDOUT:
24 a ZZZ c d
25 ## END
26
27 #### pass assoc array by reference
28 show_value() {
29 local -n array_name=$1
30 local idx=$2
31 echo "${array_name[$idx]}"
32 }
33 days=([monday]=eggs [tuesday]=bread [sunday]=jam)
34 show_value days sunday
35 ## stdout: jam
36 ## BUG mksh stdout: [monday]=eggs
37 # mksh note: it coerces "days" to 0? Horrible.
38
39 #### pass local array by reference, relying on DYNAMIC SCOPING
40 show_value() {
41 local -n array_name=$1
42 local idx=$2
43 echo "${array_name[$idx]}"
44 }
45 caller() {
46 local shadock=(ga bu zo meu)
47 show_value shadock 2
48 }
49 caller
50 ## stdout: zo
51 # mksh appears not to have local arrays!
52 ## BUG mksh stdout-json: ""
53 ## BUG mksh status: 1
54
55
56 #### flag -n and +n
57 x=foo
58
59 ref=x
60
61 echo ref=$ref
62
63 typeset -n ref
64 echo ref=$ref
65
66 # mutate underlying var
67 x=bar
68 echo ref=$ref
69
70 typeset +n ref
71 echo ref=$ref
72
73 ## STDOUT:
74 ref=x
75 ref=foo
76 ref=bar
77 ref=x
78 ## END
79
80 #### mutating through nameref: ref=
81 x=XX
82 y=YY
83
84 ref=x
85 ref=y
86 echo 1 ref=$ref
87
88 # now it's a reference
89 typeset -n ref
90
91 echo 2 ref=$ref # prints YY
92
93 ref=XXXX
94 echo 3 ref=$ref # it actually prints y, which is XXXX
95
96 # now Y is mutated!
97 echo 4 y=$y
98
99 ## STDOUT:
100 1 ref=y
101 2 ref=YY
102 3 ref=XXXX
103 4 y=XXXX
104 ## END
105
106
107 #### flag -n combined ${!ref} -- bash INVERTS
108 foo=FOO # should NOT use this
109
110 x=foo
111 ref=x
112
113 echo ref=$ref
114 echo "!ref=${!ref}"
115
116 echo 'NOW A NAMEREF'
117
118 typeset -n ref
119 echo ref=$ref
120 echo "!ref=${!ref}"
121
122 ## STDOUT:
123 ref=x
124 !ref=foo
125 NOW A NAMEREF
126 ref=foo
127 !ref=x
128 ## END
129 ## N-I mksh STDOUT:
130 ref=x
131 !ref=ref
132 NOW A NAMEREF
133 ref=foo
134 !ref=x
135 ## END
136
137 #### named ref with $# doesn't work
138 set -- one two three
139
140 ref='#'
141 echo ref=$ref
142 typeset -n ref
143 echo ref=$ref
144
145 ## STDOUT:
146 ref=#
147 ref=#
148 ## END
149
150 # mksh does respect it!! Gah.
151 ## OK mksh STDOUT:
152 ref=#
153 ref=3
154 ## END
155
156
157 #### named ref with $# and shopt -s strict_nameref
158 shopt -s strict_nameref
159
160 ref='#'
161 echo ref=$ref
162 typeset -n ref
163 echo ref=$ref
164 ## STDOUT:
165 ref=#
166 ## END
167 ## status: 1
168 ## N-I bash status: 0
169 ## N-I bash STDOUT:
170 ref=#
171 ref=#
172 ## END
173 ## N-I mksh status: 0
174 ## N-I mksh STDOUT:
175 ref=#
176 ref=0
177 ## END
178
179 #### named ref with 1 $1 etc.
180 set -- one two three
181
182 x=X
183
184 ref='1'
185 echo ref=$ref
186 typeset -n ref
187 echo ref=$ref
188
189 # BUG: This is really assigning '1', which is INVALID
190 # with strict_nameref that degrades!!!
191 ref2='$1'
192 echo ref2=$ref2
193 typeset -n ref2
194 echo ref2=$ref2
195
196 x=foo
197
198 ref3='x'
199 echo ref3=$ref3
200 typeset -n ref3
201 echo ref3=$ref3
202
203 ## STDOUT:
204 ref=1
205 ref=1
206 ref2=$1
207 ref2=$1
208 ref3=x
209 ref3=foo
210 ## END
211 ## BUG mksh status: 1
212 ## BUG mksh STDOUT:
213 ref=1
214 ref=one
215 ref2=$1
216 ## END
217
218 #### assign to invalid ref
219 ref=1 # mksh makes this READ-ONLY! Because it's not valid.
220
221 echo ref=$ref
222 typeset -n ref
223 echo ref=$ref
224
225 ref=foo
226 echo ref=$ref
227 ## STDOUT:
228 ref=1
229 ref=1
230 ref=foo
231 ## END
232 ## OK mksh status: 2
233 ## OK mksh STDOUT:
234 ref=1
235 ref=
236 ## END
237
238 #### assign to invalid ref with strict_nameref
239 case $SH in *bash|*mksh) exit ;; esac
240
241 shopt -s strict_nameref
242
243 ref=1
244
245 echo ref=$ref
246 typeset -n ref
247 echo ref=$ref
248
249 ref=foo
250 echo ref=$ref
251 ## status: 1
252 ## STDOUT:
253 ref=1
254 ## END
255 ## N-I bash/mksh status: 0
256 ## N-I bash/mksh stdout-json: ""
257
258 #### name ref on Undef cell
259 typeset -n ref
260
261 # This is technically incorrect: an undefined name shouldn't evaluate to empty
262 # string. mksh doesn't allow it.
263 echo ref=$ref
264
265 echo nounset
266 set -o nounset
267 echo ref=$ref
268 ## status: 1
269 ## STDOUT:
270 ref=
271 nounset
272 ## END
273 ## OK mksh stdout-json: ""
274
275 #### assign to empty nameref and invalid nameref
276 typeset -n ref
277 echo ref=$ref
278
279 # this is a no-op in bash, should be stricter
280 ref=x
281 echo ref=$ref
282
283 typeset -n ref2=undef
284 echo ref2=$ref2
285 ref2=x
286 echo ref2=$ref2
287
288 ## STDOUT:
289 ref=
290 ref=
291 ref2=
292 ref2=x
293 ## END
294
295 # mksh gives a good error: empty nameref target
296 ## OK mksh status: 1
297 ## OK mksh stdout-json: ""
298
299 #### -n attribute before it has a value
300 typeset -n ref
301
302 echo ref=$ref
303
304 # Now that it's a string, it still has the -n attribute
305 x=XX
306 ref=x
307 echo ref=$ref
308
309 ## STDOUT:
310 ref=
311 ref=XX
312 ## END
313 ## N-I mksh status: 1
314 ## N-I mksh stdout-json: ""
315
316 #### -n attribute on array is hard error, not a warning
317 x=X
318 typeset -n ref #=x
319 echo hi
320
321 # bash prints warning: REMOVES the nameref attribute here!
322 ref=(x y)
323 echo ref=$ref
324
325 ## status: 1
326 ## STDOUT:
327 hi
328 ## END
329 ## N-I mksh status: 1
330 ## N-I mksh stdout-json: ""
331 ## BUG bash status: 0
332 ## BUG bash STDOUT:
333 hi
334 ref=x
335 ## END
336
337 #### exported nameref
338 x=foo
339 typeset -n -x ref=x
340
341 # hm bash ignores it but mksh doesn't. maybe disallow it.
342 printenv.py x ref
343 echo ---
344 export x
345 printenv.py x ref
346 ## STDOUT:
347 None
348 x
349 ---
350 foo
351 x
352 ## END
353 ## OK mksh STDOUT:
354 None
355 None
356 ---
357 foo
358 None
359 ## END
360
361
362 #### readonly nameref doesn't prevent assigning through it
363
364 # hm bash also ignores -r when -n is set
365
366 x=XX
367 typeset -n -r ref=x
368
369 echo ref=$ref
370
371 # it feels like I shouldn't be able to mutate this?
372 ref=XXXX
373 echo ref=$ref
374
375 x=X
376 echo x=$x
377
378 ## STDOUT:
379 ref=XX
380 ref=XXXX
381 x=X
382 ## END
383
384 #### readonly var can't be assigned through nameref
385
386 x=X
387 typeset -n -r ref=x
388
389 echo ref=$ref
390
391 # it feels like I shouldn't be able to mutate this?
392 ref=XX
393 echo ref=$ref
394
395 # now the underling variable is immutable
396 typeset -r x
397
398 ref=XXX
399 echo ref=$ref
400 echo x=$x
401
402 ## status: 1
403 ## OK mksh status: 2
404 ## STDOUT:
405 ref=X
406 ref=XX
407 ## END
408
409 ## OK bash status: 0
410 ## OK bash STDOUT:
411 ref=X
412 ref=XX
413 ref=XX
414 x=XX
415 ## END
416
417 #### unset nameref
418 x=X
419 typeset -n ref=x
420 echo ref=$ref
421
422 # this works
423 unset ref
424 echo ref=$ref
425 echo x=$x
426
427 ## STDOUT:
428 ref=X
429 ref=
430 x=
431 ## END
432
433 #### Chain of namerefs
434 x=foo
435 typeset -n ref=x
436 typeset -n ref_to_ref=ref
437 echo ref_to_ref=$ref_to_ref
438 echo ref=$ref
439 ## STDOUT:
440 ref_to_ref=foo
441 ref=foo
442 ## END
443
444 #### Mutually recursive namerefs detected on READ
445 typeset -n ref1=ref2
446 typeset -n ref2=ref1
447 echo defined
448 echo ref1=$ref1
449 echo ref2=$ref1
450 ## status: 1
451 ## STDOUT:
452 defined
453 ## END
454 ## OK mksh stdout-json: ""
455 ## BUG bash status: 0
456 ## BUG bash STDOUT:
457 defined
458 ref1=
459 ref2=
460 ## END
461
462 #### Mutually recursive namerefs detected on WRITE
463 typeset -n ref1=ref2
464 typeset -n ref2=ref1 # not detected here
465 echo defined $?
466 ref1=z # detected here
467 echo mutated $?
468 ## status: 1
469 ## STDOUT:
470 defined 0
471 ## END
472 ## OK mksh stdout-json: ""
473 ## BUG bash status: 0
474 ## BUG bash STDOUT:
475 defined 0
476 mutated 1
477 ## END
478
479 #### Dynamic scope with namerefs
480
481 f3() {
482 local -n ref=$1
483 ref=x
484 }
485
486 f2() {
487 f3 "$@"
488 }
489
490 f1() {
491 local F1=F1
492 echo F1=$F1
493 f2 F1
494 echo F1=$F1
495 }
496 f1
497
498 ## STDOUT:
499 F1=F1
500 F1=x
501 ## END
502
503
504 #### change reference itself
505 x=XX
506 y=YY
507 typeset -n ref=x
508 echo ref=$ref
509 echo x=$x
510 echo y=$y
511
512 echo ----
513 typeset -n ref=y
514 echo ref=$ref
515 echo x=$x
516 echo y=$y
517 echo ----
518 ref=z
519 echo ref=$ref
520 echo x=$x
521 echo y=$y
522
523 ## STDOUT:
524 ref=XX
525 x=XX
526 y=YY
527 ----
528 ref=YY
529 x=XX
530 y=YY
531 ----
532 ref=z
533 x=XX
534 y=z
535 ## END
536
537 #### a[2] in nameref
538
539 typeset -n ref='a[2]'
540 a=(zero one two three)
541 echo ref=$ref
542 ## STDOUT:
543 ref=two
544 ## END
545
546 #### a[expr] in nameref
547
548 # this confuses code and data
549 typeset -n ref='a[$(echo 2) + 1]'
550 a=(zero one two three)
551 echo ref=$ref
552 ## STDOUT:
553 ref=three
554 ## END
555
556 #### a[@] in nameref
557
558 # this confuses code and data
559 typeset -n ref='a[@]'
560 a=('A B' C)
561 argv.py ref "$ref" # READ through ref works
562 ref=(X Y Z) # WRITE through doesn't work
563 echo status=$?
564 argv.py 'ref[@]' "${ref[@]}"
565 argv.py ref "$ref" # JOINING mangles the array?
566 argv.py 'a[@]' "${a[@]}"
567 ## STDOUT:
568 ['ref', 'A B C']
569 status=1
570 ['ref[@]']
571 ['ref', 'A B C']
572 ['a[@]', 'A B', 'C']
573 ## END
574 ## OK mksh status: 1
575 ## OK mksh stdout-json: ""
576
577 #### mutate through nameref: ref[0]=
578
579 # This is DIFFERENT than the nameref itself being 'array[0]' !
580
581 array=(X Y Z)
582 typeset -n ref=array
583 ref[0]=xx
584 echo ${array[@]}
585 ## STDOUT:
586 xx Y Z
587 ## END
588
589 #### bad mutation through nameref: ref[0]= where ref is array[0]
590 array=(X Y Z)
591 typeset -n ref='array[0]'
592 ref[0]=foo # error in bash: 'array[0]': not a valid identifier
593 echo status=$?
594 echo ${array[@]}
595 ## STDOUT:
596 status=1
597 X Y Z
598 ## END
599 ## BUG mksh STDOUT:
600 status=0
601 foo Y Z
602 ## END
603
604 #### @ in nameref isn't supported, unlike in ${!ref}
605
606 set -- A B
607 typeset -n ref='@' # bash gives an error here
608 echo status=$?
609
610 echo ref=$ref # bash doesn't give an error here
611 echo status=$?
612 ## status: 1
613 ## stdout-json: ""
614 ## OK bash status: 0
615 ## OK bash STDOUT:
616 status=1
617 ref=
618 status=0
619 ## END
620
621 #### Unquoted assoc reference on RHS
622 typeset -A bashup_ev_r
623 bashup_ev_r['foo']=bar
624
625 p() {
626 local s=foo
627 local -n e=bashup_ev["$s"] f=bashup_ev_r["$s"]
628 # Different!
629 #local e=bashup_ev["$s"] f=bashup_ev_r["$s"]
630 argv.py "$f"
631 }
632 p
633 ## STDOUT:
634 ['bar']
635 ## END
636 ## N-I mksh stdout-json: ""
637 ## N-I mksh status: 1