OILS / spec / var-op-bash.test.sh View on Github | oils.pub

538 lines, 297 significant
1## compare_shells: bash
2## oils_failures_allowed: 8
3
4#### Lower Case with , and ,,
5x='ABC DEF'
6echo ${x,}
7echo ${x,,}
8echo empty=${empty,}
9echo empty=${empty,,}
10## STDOUT:
11aBC DEF
12abc def
13empty=
14empty=
15## END
16
17#### Upper Case with ^ and ^^
18x='abc def'
19echo ${x^}
20echo ${x^^}
21echo empty=${empty^}
22echo empty=${empty^^}
23## STDOUT:
24Abc def
25ABC DEF
26empty=
27empty=
28## END
29
30#### Case folding - Unicode characters
31
32# https://www.utf8-chartable.de/unicode-utf8-table.pl
33
34x=$'\u00C0\u00C8' # upper grave
35y=$'\u00E1\u00E9' # lower acute
36
37echo u ${x^}
38echo U ${x^^}
39
40echo l ${x,}
41echo L ${x,,}
42
43echo u ${y^}
44echo U ${y^^}
45
46echo l ${y,}
47echo L ${y,,}
48
49## STDOUT:
50u ÀÈ
51U ÀÈ
52l àÈ
53L àè
54u Áé
55U ÁÉ
56l áé
57L áé
58## END
59
60#### Case folding - multi code point
61
62echo shell
63small=$'\u00DF'
64echo u ${small^}
65echo U ${small^^}
66
67echo l ${small,}
68echo L ${small,,}
69echo
70
71echo python2
72python2 -c '
73small = u"\u00DF"
74print(small.upper().encode("utf-8"))
75print(small.lower().encode("utf-8"))
76'
77echo
78
79# Not in the container images, but python 3 DOES support it!
80# This is moved to demo/survey-case-fold.sh
81
82if false; then
83echo python3
84python3 -c '
85import sys
86small = u"\u00DF"
87sys.stdout.buffer.write(small.upper().encode("utf-8") + b"\n")
88sys.stdout.buffer.write(small.lower().encode("utf-8") + b"\n")
89'
90fi
91
92if false; then
93 # Yes, supported
94 echo node.js
95
96 nodejs -e '
97 var small = "\u00DF"
98 console.log(small.toUpperCase())
99 console.log(small.toLowerCase())
100 '
101fi
102
103## STDOUT:
104## END
105## BUG bash STDOUT:
106shell
107u ß
108U ß
109l ß
110L ß
111
112python2
113ß
114ß
115
116## END
117
118#### Case folding that depends on locale (not enabled, requires Turkish locale)
119
120# Hm this works in demo/survey-case-fold.sh
121# Is this a bash 4.4 thing?
122
123#export LANG='tr_TR.UTF-8'
124#echo $LANG
125
126x='i'
127
128echo u ${x^}
129echo U ${x^^}
130
131echo l ${x,}
132echo L ${x,,}
133
134## OK bash/osh STDOUT:
135u I
136U I
137l i
138L i
139## END
140
141#### Lower Case with constant string (VERY WEIRD)
142x='AAA ABC DEF'
143echo ${x,A}
144echo ${x,,A} # replaces every A only?
145## STDOUT:
146aAA ABC DEF
147aaa aBC DEF
148## END
149
150#### Lower Case glob
151
152# Hm with C.UTF-8, this does no case folding?
153export LC_ALL=en_US.UTF-8
154
155x='ABC DEF'
156echo ${x,[d-f]}
157echo ${x,,[d-f]} # bash 4.4 fixed in bash 5.2.21
158## STDOUT:
159ABC DEF
160ABC DEF
161## END
162
163#### ${x@u} U L - upper / lower case (bash 5.1 feature)
164
165# https://www.gnu.org/software/bash/manual/html_node/Shell-Parameter-Expansion.html
166
167x='abc DEF'
168
169echo "${x@u}"
170
171echo "${x@U}"
172
173echo "${x@L}"
174
175## STDOUT:
176Abc DEF
177ABC DEF
178abc def
179## END
180
181
182#### ${x@Q}
183x="FOO'BAR spam\"eggs"
184eval "new=${x@Q}"
185test "$x" = "$new" && echo OK
186## STDOUT:
187OK
188## END
189
190#### ${array@Q} and ${array[@]@Q}
191array=(x 'y\nz')
192echo ${array[@]@Q}
193echo ${array@Q}
194echo ${array@Q}
195## STDOUT:
196'x' 'y\nz'
197'x'
198'x'
199## END
200## OK osh STDOUT:
201x $'y\\nz'
202x
203x
204## END
205
206#### ${!prefix@} ${!prefix*} yields sorted array of var names
207ZOO=zoo
208ZIP=zip
209ZOOM='one two'
210Z='three four'
211
212z=lower
213
214argv.py ${!Z*}
215argv.py ${!Z@}
216argv.py "${!Z*}"
217argv.py "${!Z@}"
218for i in 1 2; do argv.py ${!Z*} ; done
219for i in 1 2; do argv.py ${!Z@} ; done
220for i in 1 2; do argv.py "${!Z*}"; done
221for i in 1 2; do argv.py "${!Z@}"; done
222## STDOUT:
223['Z', 'ZIP', 'ZOO', 'ZOOM']
224['Z', 'ZIP', 'ZOO', 'ZOOM']
225['Z ZIP ZOO ZOOM']
226['Z', 'ZIP', 'ZOO', 'ZOOM']
227['Z', 'ZIP', 'ZOO', 'ZOOM']
228['Z', 'ZIP', 'ZOO', 'ZOOM']
229['Z', 'ZIP', 'ZOO', 'ZOOM']
230['Z', 'ZIP', 'ZOO', 'ZOOM']
231['Z ZIP ZOO ZOOM']
232['Z ZIP ZOO ZOOM']
233['Z', 'ZIP', 'ZOO', 'ZOOM']
234['Z', 'ZIP', 'ZOO', 'ZOOM']
235## END
236
237#### ${!prefix@} matches var name (regression)
238hello1=1 hello2=2 hello3=3
239echo ${!hello@}
240hello=()
241echo ${!hello@}
242## STDOUT:
243hello1 hello2 hello3
244hello hello1 hello2 hello3
245## END
246
247#### ${var@a} for attributes
248array=(one two)
249echo ${array@a}
250declare -r array=(one two)
251echo ${array@a}
252declare -rx PYTHONPATH=hi
253echo ${PYTHONPATH@a}
254
255# bash and osh differ here
256#declare -rxn x=z
257#echo ${x@a}
258## STDOUT:
259a
260ar
261rx
262## END
263
264#### ${var@a} error conditions
265echo [${?@a}]
266## STDOUT:
267[]
268## END
269
270#### undef and @P @Q @a
271$SH -c 'echo ${undef@P}'
272echo status=$?
273$SH -c 'echo ${undef@Q}'
274echo status=$?
275$SH -c 'echo ${undef@a}'
276echo status=$?
277## STDOUT:
278
279status=0
280
281status=0
282
283status=0
284## END
285
286
287#### argv array and @P @Q @a
288$SH -c 'echo ${@@P}' dummy a b c
289echo status=$?
290$SH -c 'echo ${@@Q}' dummy a 'b\nc'
291echo status=$?
292$SH -c 'echo ${@@a}' dummy a b c
293echo status=$?
294## STDOUT:
295a b c
296status=0
297'a' 'b\nc'
298status=0
299
300status=0
301## END
302
303#### assoc array and @P @Q @a
304
305# note: "y z" causes a bug!
306$SH -c 'declare -A A=(["x"]="y"); echo ${A@P} - ${A[@]@P}'
307echo status=$?
308
309# note: "y z" causes a bug!
310$SH -c 'declare -A A=(["x"]="y"); echo ${A@Q} - ${A[@]@Q}' | sed 's/^- y$/- '\''y'\''/'
311echo status=$?
312
313$SH -c 'declare -A A=(["x"]=y); echo ${A@a} - ${A[@]@a}'
314echo status=$?
315## STDOUT:
316- y
317status=0
318- 'y'
319status=0
320A - A
321status=0
322## END
323
324#### ${!var[@]@X}
325# note: "y z" causes a bug!
326$SH -c 'declare -A A=(["x"]="y"); echo ${!A[@]@P}'
327if test $? -ne 0; then echo fail; fi
328
329# note: "y z" causes a bug!
330$SH -c 'declare -A A=(["x y"]="y"); echo ${!A[@]@Q}'
331if test $? -ne 0; then echo fail; fi
332
333$SH -c 'declare -A A=(["x"]=y); echo ${!A[@]@a}'
334if test $? -ne 0; then echo fail; fi
335# STDOUT:
336
337
338
339# END
340## OK osh STDOUT:
341fail
342'x y'
343a
344## END
345
346#### ${#var@X} is a parse error
347# note: "y z" causes a bug!
348$SH -c 'declare -A A=(["x"]="y"); echo ${#A[@]@P}'
349if test $? -ne 0; then echo fail; fi
350
351# note: "y z" causes a bug!
352$SH -c 'declare -A A=(["x"]="y"); echo ${#A[@]@Q}'
353if test $? -ne 0; then echo fail; fi
354
355$SH -c 'declare -A A=(["x"]=y); echo ${#A[@]@a}'
356if test $? -ne 0; then echo fail; fi
357## STDOUT:
358fail
359fail
360fail
361## END
362
363#### ${!A@a} and ${!A[@]@a}
364declare -A A=(["x"]=y)
365echo x=${!A[@]@a}
366echo invalid=${!A@a}
367
368# OSH prints 'a' for indexed array because the AssocArray with ! turns into
369# it. Disallowing it would be the other reasonable behavior.
370
371## status: 1
372## STDOUT:
373x=
374## END
375
376# Bash succeeds with ${!A@a}, which references the variable named as $A (i.e.,
377# ''). This must be a Bash bug since the behavior is inconsistent with the
378# fact that ${!undef@a} and ${!empty@a} fail.
379
380## BUG bash status: 0
381## BUG bash STDOUT:
382x=
383invalid=
384## END
385
386#### undef vs. empty string in var ops
387
388empty=''
389x=x
390
391echo ${x@Q} ${empty@Q} ${undef@Q} ${x@Q}
392
393echo ${x@K} ${empty@K} ${undef@K} ${x@K}
394
395echo ${x@k} ${empty@k} ${undef@k} ${x@k}
396
397echo ${x@A} ${empty@A} ${undef@A} ${x@A}
398
399declare -r x
400echo ${x@a} ${empty@a} ${undef@a} ${x@a}
401
402# x x
403#echo ${x@E} ${empty@E} ${undef@E} ${x@E}
404# x x
405#echo ${x@P} ${empty@P} ${undef@P} ${x@P}
406
407## STDOUT:
408'x' '' 'x'
409'x' '' 'x'
410'x' '' 'x'
411x='x' empty='' x='x'
412r r
413## END
414
415#### -o nounset with var ops
416
417set -u
418(echo ${undef@Q}); echo "stat: $?"
419(echo ${undef@P}); echo "stat: $?"
420(echo ${undef@a}); echo "stat: $?"
421
422## STDOUT:
423stat: 1
424stat: 1
425stat: 1
426## END
427
428
429#### ${a[0]@a} and ${a@a}
430
431a=(1 2 3)
432echo "attr = '${a[0]@a}'"
433echo "attr = '${a@a}'"
434
435## STDOUT:
436attr = 'a'
437attr = 'a'
438## END
439
440
441#### ${!r@a} with r='a[0]' (attribute for indirect expansion of an array element)
442
443a=(1 2 3)
444r='a'
445echo ${!r@a}
446r='a[0]'
447echo ${!r@a}
448
449declare -A d=([0]=foo [1]=bar)
450r='d'
451echo ${!r@a}
452r='d[0]'
453echo ${!r@a}
454
455## STDOUT:
456a
457a
458A
459A
460## END
461
462
463#### Array expansion with nullary var ops
464
465declare -a a=({1..9})
466declare -A A=(['a']=hello ['b']=world ['c']=osh ['d']=ysh)
467
468echo "@Q"
469argv.py "${a[@]@Q}"
470argv.py "${a[*]@Q}"
471argv.py "${A[@]@Q}"
472argv.py "${A[*]@Q}"
473argv.py "${u[@]@Q}"
474argv.py "${u[*]@Q}"
475
476echo "@P"
477argv.py "${a[@]@P}"
478argv.py "${a[*]@P}"
479argv.py "${A[@]@P}"
480argv.py "${A[*]@P}"
481argv.py "${u[@]@P}"
482argv.py "${u[*]@P}"
483
484echo "@a"
485argv.py "${a[@]@a}"
486argv.py "${a[*]@a}"
487argv.py "${A[@]@a}"
488argv.py "${A[*]@a}"
489argv.py "${u[@]@a}"
490argv.py "${u[*]@a}"
491
492## STDOUT:
493@Q
494['1', '2', '3', '4', '5', '6', '7', '8', '9']
495['1 2 3 4 5 6 7 8 9']
496['hello', 'world', 'osh', 'ysh']
497['hello world osh ysh']
498[]
499['']
500@P
501['1', '2', '3', '4', '5', '6', '7', '8', '9']
502['1 2 3 4 5 6 7 8 9']
503['hello', 'world', 'osh', 'ysh']
504['hello world osh ysh']
505[]
506['']
507@a
508['a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a']
509['a a a a a a a a a']
510['A', 'A', 'A', 'A']
511['A A A A']
512[]
513['']
514## END
515
516## OK bash STDOUT:
517@Q
518["'1'", "'2'", "'3'", "'4'", "'5'", "'6'", "'7'", "'8'", "'9'"]
519["'1' '2' '3' '4' '5' '6' '7' '8' '9'"]
520["'ysh'", "'osh'", "'world'", "'hello'"]
521["'ysh' 'osh' 'world' 'hello'"]
522[]
523['']
524@P
525['1', '2', '3', '4', '5', '6', '7', '8', '9']
526['1 2 3 4 5 6 7 8 9']
527['ysh', 'osh', 'world', 'hello']
528['ysh osh world hello']
529[]
530['']
531@a
532['a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a']
533['a a a a a a a a a']
534['A', 'A', 'A', 'A']
535['A A A A']
536[]
537['']
538## END