1 |
|
2 | #### command sub $(echo hi)
|
3 | var x = $(echo hi)
|
4 | var y = $(echo '')
|
5 | # Make sure we can operate on these values
|
6 | echo x=${x:-default} y=${y:-default}
|
7 | ## STDOUT:
|
8 | x=hi y=default
|
9 | ## END
|
10 |
|
11 | #### shell array %(a 'b c')
|
12 | shopt -s parse_at
|
13 | var x = %(a 'b c')
|
14 | var empty = %()
|
15 | argv.py / @x @empty /
|
16 |
|
17 | ## STDOUT:
|
18 | ['/', 'a', 'b c', '/']
|
19 | ## END
|
20 |
|
21 | #### empty array and simple_word_eval (regression test)
|
22 | shopt -s parse_at simple_word_eval
|
23 | var empty = :| |
|
24 | echo len=$[len(empty)]
|
25 | argv.py / @empty /
|
26 |
|
27 | ## STDOUT:
|
28 | len=0
|
29 | ['/', '/']
|
30 | ## END
|
31 |
|
32 | #### Empty array and assignment builtin (regression)
|
33 | # Bug happens with shell arrays too
|
34 | empty=()
|
35 | declare z=1 "${empty[@]}"
|
36 | echo z=$z
|
37 | ## STDOUT:
|
38 | z=1
|
39 | ## END
|
40 |
|
41 | #### Shell arrays support tilde detection, static globbing, brace detection
|
42 | shopt -s parse_at simple_word_eval
|
43 | touch {foo,bar}.py
|
44 | HOME=/home/bob
|
45 | no_dynamic_glob='*.py'
|
46 |
|
47 | var x = %(~/src *.py {andy,bob}@example.com $no_dynamic_glob)
|
48 | argv.py @x
|
49 | ## STDOUT:
|
50 | ['/home/bob/src', 'bar.py', 'foo.py', 'andy@example.com', 'bob@example.com', '*.py']
|
51 | ## END
|
52 |
|
53 | #### Set $HOME using 'var' (i.e. Oil string var in word evaluator)
|
54 | var HOME = "foo"
|
55 | echo $HOME
|
56 | echo ~
|
57 | ## STDOUT:
|
58 | foo
|
59 | foo
|
60 | ## END
|
61 |
|
62 | #### Use shell var in Oil expression
|
63 | x='abc'
|
64 | var length = len(x) # length in BYTES, unlike ${#x}
|
65 | echo $length
|
66 | ## STDOUT:
|
67 | 3
|
68 | ## END
|
69 |
|
70 | #### Length doesn't apply to BashArray
|
71 | x=(a b c)
|
72 | x[10]=A
|
73 | x[20]=B
|
74 |
|
75 | # shell style: length is 5
|
76 | echo shell=${#x[@]}
|
77 |
|
78 | # Length could be 20, but we may change the representation to Dict[int, str]
|
79 | echo ysh=$[len(x)]
|
80 |
|
81 | ## status: 3
|
82 | ## STDOUT:
|
83 | shell=5
|
84 | ## END
|
85 |
|
86 | #### $[len(x)] inside strings
|
87 | var s = "abc"
|
88 | echo -$[len(s)]-
|
89 |
|
90 | # This already has a meaning ...
|
91 | #echo "-$len(x)-"
|
92 | #echo "-${len}(x)-"
|
93 |
|
94 | ## STDOUT:
|
95 | -3-
|
96 | ## END
|
97 |
|
98 | #### Func with multiple args in multiple contexts
|
99 | shopt --set ysh:upgrade # needed for math.ysh
|
100 |
|
101 | source --builtin math.ysh
|
102 |
|
103 | var x = max(1+2, 3+4)
|
104 | echo $x $[max(1+2, 3+4)]
|
105 |
|
106 | ## STDOUT:
|
107 | 7 7
|
108 | ## END
|
109 |
|
110 |
|
111 | #### Trailing Comma in Param list
|
112 | shopt --set ysh:upgrade # needed for math.ysh
|
113 |
|
114 | source --builtin math.ysh
|
115 |
|
116 | var x = max(1+2, 3+4,)
|
117 | echo $x $[max(1+2, 3+4,)]
|
118 |
|
119 | ## STDOUT:
|
120 | 7 7
|
121 | ## END
|
122 |
|
123 | #### nested expr contexts
|
124 | var s = "123"
|
125 |
|
126 | # lex_mode_e.ShCommand -> Expr -> ShCommand -> Expr
|
127 | var x = $(echo $'len\n' $[len(s)])
|
128 | echo $x
|
129 | ## STDOUT:
|
130 | len 3
|
131 | ## END
|
132 |
|
133 |
|
134 | # TODO:
|
135 | # - test keyword args
|
136 | # - test splatting *args, **kwargs
|
137 | # - Multiline parsing
|
138 | #
|
139 | # var x = max(
|
140 | # 1+2,
|
141 | # 3+4,
|
142 | # )
|
143 | # echo $x $max(
|
144 | # 1+2,
|
145 | # 3+4,
|
146 | # )
|
147 |
|
148 | #### YSH var used with shell arithmetic
|
149 | var w = "3"
|
150 | echo lt=$(( w < 4 ))
|
151 | echo gt=$(( w > 4 ))
|
152 |
|
153 | var z = 3
|
154 | echo lt=$(( z < 4 ))
|
155 | echo gt=$(( z > 4 ))
|
156 | ## STDOUT:
|
157 | lt=1
|
158 | gt=0
|
159 | lt=1
|
160 | gt=0
|
161 | ## END
|
162 |
|
163 | #### Parse { var x = 42 }
|
164 | shopt -s oil:upgrade
|
165 | g() { var x = 42 }
|
166 |
|
167 | var x = 1
|
168 | f() { var x = 42; setvar x = 43 }
|
169 | f
|
170 | echo x=$x
|
171 | ## STDOUT:
|
172 | x=1
|
173 | ## END
|
174 |
|
175 | #### double quoted
|
176 | var foo = "bar"
|
177 | var x = "-$foo-${foo}-${undef:-default}-"
|
178 | echo $x
|
179 | ## STDOUT:
|
180 | -bar-bar-default-
|
181 | ## END
|
182 |
|
183 | #### double quoted respects strict_array
|
184 | shopt -s strict:all
|
185 | declare -a a=(one two three)
|
186 | var x = "-${a[@]}-"
|
187 | echo $x
|
188 | ## status: 1
|
189 | ## stdout-json: ""
|
190 |
|
191 | #### simple var sub $name $0 $1 $? etc.
|
192 | ( exit 42 )
|
193 | var status = $?
|
194 | echo status=$status
|
195 |
|
196 | set -- a b c
|
197 | var one = $1
|
198 | var two = $2
|
199 | echo $one $two
|
200 |
|
201 | var named = "$one" # equivalent to 'one'
|
202 | echo named=$named
|
203 |
|
204 | ## STDOUT:
|
205 | status=42
|
206 | a b
|
207 | named=a
|
208 | ## END
|
209 |
|
210 | #### braced var sub ${x:-default}
|
211 |
|
212 | # without double quotes
|
213 |
|
214 | var b = ${foo:-default}
|
215 | echo $b
|
216 | var c = ${bar:-"-$b-"}
|
217 | echo $c
|
218 |
|
219 | var d = "${bar:-"-$c-"}" # another one
|
220 | echo $d
|
221 |
|
222 | ## STDOUT:
|
223 | default
|
224 | -default-
|
225 | --default--
|
226 | ## END
|
227 |
|
228 | #### braced var sub respects strict_array
|
229 | set -- a b c
|
230 | var x = ${undef:-"$@"}
|
231 | echo $x
|
232 | shopt -s strict_array
|
233 | setvar x = ${undef:-"$@"}
|
234 | echo $x
|
235 | ## status: 1
|
236 | ## STDOUT:
|
237 | a b c
|
238 | ## END
|
239 |
|
240 |
|
241 | #### null / true / false
|
242 | shopt -s oil:upgrade
|
243 | var n = null
|
244 | if (n) {
|
245 | echo yes
|
246 | } else {
|
247 | echo no
|
248 | }
|
249 | var t = true
|
250 | if (t) {
|
251 | echo yes
|
252 | } else {
|
253 | echo no
|
254 | }
|
255 | var f = false
|
256 | if (f) {
|
257 | echo yes
|
258 | } else {
|
259 | echo no
|
260 | }
|
261 | ## STDOUT:
|
262 | no
|
263 | yes
|
264 | no
|
265 | ## END
|
266 |
|
267 | #### Integer literals
|
268 | var d = 123
|
269 | var b = 0b11
|
270 | var o = 0o123
|
271 | var h = 0xff
|
272 | echo $d $b $o $h
|
273 | ## STDOUT:
|
274 | 123 3 83 255
|
275 | ## END
|
276 |
|
277 | #### Integer literals with underscores
|
278 | const dec = 65_536
|
279 | const bin = 0b0001_0101
|
280 | const oct = 0o001_755
|
281 | const hex = 0x0001_000f
|
282 |
|
283 | echo SHELL
|
284 | echo $dec
|
285 | echo $bin
|
286 | echo $oct
|
287 | echo $hex
|
288 | const x = 1_1 + 0b1_1 + 0o1_1 + 0x1_1
|
289 | echo sum $x
|
290 |
|
291 | # This works under Python 3.6, but the continuous build has earlier versions
|
292 | if false; then
|
293 | echo ---
|
294 | echo PYTHON
|
295 |
|
296 | python3 -c '
|
297 | print(65_536)
|
298 | print(0b0001_0101)
|
299 | print(0o001_755)
|
300 | print(0x0001_000f)
|
301 |
|
302 | # Weird syntax
|
303 | print("sum", 1_1 + 0b1_1 + 0o1_1 + 0x1_1)
|
304 | '
|
305 | fi
|
306 |
|
307 | ## STDOUT:
|
308 | SHELL
|
309 | 65536
|
310 | 21
|
311 | 1005
|
312 | 65551
|
313 | sum 40
|
314 | ## END
|
315 |
|
316 | #### Backslash char literal (is an integer)
|
317 | const newline = \n
|
318 | const backslash = \\
|
319 | const sq = \'
|
320 | const dq = \"
|
321 | echo "$newline $backslash $sq $dq"
|
322 | ## STDOUT:
|
323 | 10 92 39 34
|
324 | ## END
|
325 |
|
326 | #### \u{3bc} is char literal
|
327 | shopt -s oil:all
|
328 |
|
329 | var mu = \u{3bc}
|
330 | if (mu === 0x3bc) { # this is the same!
|
331 | echo 'yes'
|
332 | }
|
333 | echo "mu $mu"
|
334 | ## STDOUT:
|
335 | yes
|
336 | mu 956
|
337 | ## END
|
338 |
|
339 | #### Exponentiation with **
|
340 | var x = 2**3
|
341 | echo $x
|
342 |
|
343 | var y = 2.0 ** 3.0 # NOT SUPPORTED
|
344 | echo 'should not get here'
|
345 |
|
346 | ## status: 3
|
347 | ## STDOUT:
|
348 | 8
|
349 | ## END
|
350 |
|
351 | #### Float Division
|
352 | pp line (5/2)
|
353 | pp line (-5/2)
|
354 | pp line (5/-2)
|
355 | pp line (-5/-2)
|
356 |
|
357 | echo ---
|
358 |
|
359 | var x = 9
|
360 | setvar x /= 2
|
361 | pp line (x)
|
362 |
|
363 | var x = -9
|
364 | setvar x /= 2
|
365 | pp line (x)
|
366 |
|
367 | var x = 9
|
368 | setvar x /= -2
|
369 | pp line (x)
|
370 |
|
371 | var x = -9
|
372 | setvar x /= -2
|
373 | pp line (x)
|
374 |
|
375 |
|
376 | ## STDOUT:
|
377 | (Float) 2.5
|
378 | (Float) -2.5
|
379 | (Float) -2.5
|
380 | (Float) 2.5
|
381 | ---
|
382 | (Float) 4.5
|
383 | (Float) -4.5
|
384 | (Float) -4.5
|
385 | (Float) 4.5
|
386 | ## END
|
387 |
|
388 | #### Integer Division (rounds toward zero)
|
389 | pp line (5//2)
|
390 | pp line (-5//2)
|
391 | pp line (5//-2)
|
392 | pp line (-5//-2)
|
393 |
|
394 | echo ---
|
395 |
|
396 | var x = 9
|
397 | setvar x //= 2
|
398 | pp line (x)
|
399 |
|
400 | var x = -9
|
401 | setvar x //= 2
|
402 | pp line (x)
|
403 |
|
404 | var x = 9
|
405 | setvar x //= -2
|
406 | pp line (x)
|
407 |
|
408 | var x = -9
|
409 | setvar x //= -2
|
410 | pp line (x)
|
411 |
|
412 | ## STDOUT:
|
413 | (Int) 2
|
414 | (Int) -2
|
415 | (Int) -2
|
416 | (Int) 2
|
417 | ---
|
418 | (Int) 4
|
419 | (Int) -4
|
420 | (Int) -4
|
421 | (Int) 4
|
422 | ## END
|
423 |
|
424 | #### % operator is remainder
|
425 | pp line ( 5 % 3)
|
426 | pp line (-5 % 3)
|
427 |
|
428 | # negative divisor illegal (tested in test/ysh-runtime-errors.sh)
|
429 | #pp line ( 5 % -3)
|
430 | #pp line (-5 % -3)
|
431 |
|
432 | var z = 10
|
433 | setvar z %= 3
|
434 | pp line (z)
|
435 |
|
436 | var z = -10
|
437 | setvar z %= 3
|
438 | pp line (z)
|
439 |
|
440 | ## STDOUT:
|
441 | (Int) 2
|
442 | (Int) -2
|
443 | (Int) 1
|
444 | (Int) -1
|
445 | ## END
|
446 |
|
447 | #### Bitwise logical
|
448 | var a = 0b0101 & 0b0011
|
449 | echo $a
|
450 | var b = 0b0101 | 0b0011
|
451 | echo $b
|
452 | var c = 0b0101 ^ 0b0011
|
453 | echo $c
|
454 | var d = ~b
|
455 | echo $d
|
456 | ## STDOUT:
|
457 | 1
|
458 | 7
|
459 | 6
|
460 | -8
|
461 | ## END
|
462 |
|
463 | #### Shift operators
|
464 | var a = 1 << 4
|
465 | echo $a
|
466 | var b = 16 >> 4
|
467 | echo $b
|
468 | ## STDOUT:
|
469 | 16
|
470 | 1
|
471 | ## END
|
472 |
|
473 | #### multiline strings, list, tuple syntax for list, etc.
|
474 | var dq = "
|
475 | dq
|
476 | 2
|
477 | "
|
478 | echo dq=$[len(dq)]
|
479 |
|
480 | var sq = '
|
481 | sq
|
482 | 2
|
483 | '
|
484 | echo sq=$[len(sq)]
|
485 |
|
486 | var mylist = [
|
487 | 1,
|
488 | 2,
|
489 | 3,
|
490 | ]
|
491 | echo mylist=$[len(mylist)]
|
492 |
|
493 | var mytuple = (1,
|
494 | 2, 3)
|
495 | echo mytuple=$[len(mytuple)]
|
496 |
|
497 | ## STDOUT:
|
498 | dq=6
|
499 | sq=6
|
500 | mylist=3
|
501 | mytuple=3
|
502 | ## END
|
503 |
|
504 | #### multiline dict
|
505 |
|
506 | # Note: a pair has to be all on one line. We could relax that but there isn't
|
507 | # a strong reason to now.
|
508 |
|
509 | var mydict = { a:1,
|
510 | b: 2,
|
511 | }
|
512 | echo mydict=$[len(mydict)]
|
513 | ## STDOUT:
|
514 | mydict=2
|
515 | ## END
|
516 |
|
517 | #### multiline array and command sub (only here docs disallowed)
|
518 | var array = %(
|
519 | one
|
520 | two
|
521 | three
|
522 | )
|
523 | echo array=$[len(array)]
|
524 |
|
525 | var comsub = $(
|
526 | echo hi
|
527 | echo bye
|
528 | )
|
529 | echo comsub=$[len(comsub)]
|
530 |
|
531 | ## STDOUT:
|
532 | array=3
|
533 | comsub=6
|
534 | ## END
|
535 |
|
536 | #### obj->method()
|
537 | var s = 'hi'
|
538 |
|
539 | # TODO: This does a bound method thing we probably don't want
|
540 | var s2 = s->upper()
|
541 | echo $s2
|
542 | ## STDOUT:
|
543 | HI
|
544 | ## END
|
545 |
|
546 | #### obj->method does NOT give you a bound method
|
547 | var s = 'hi'
|
548 | var method = s->upper
|
549 | echo $method
|
550 | ## status: 3
|
551 | ## stdout-json: ""
|
552 |
|
553 | #### d.key
|
554 | var d = {name: 'andy'}
|
555 | var x = d.name
|
556 | echo $x
|
557 | ## STDOUT:
|
558 | andy
|
559 | ## END
|
560 |
|
561 | #### a ++ b for string/list concatenation
|
562 | shopt -s parse_brace
|
563 |
|
564 | var i = 'abc'
|
565 | var j = 'de'
|
566 | var k = i ++ j
|
567 | echo string $k
|
568 |
|
569 |
|
570 | var a = [1, 2]
|
571 | var b = [3]
|
572 | var c = a ++ b
|
573 | echo list len=$[len(c)]
|
574 |
|
575 | echo ---
|
576 |
|
577 | try {
|
578 | = 'ab' ++ 3
|
579 | }
|
580 | echo Str Int $_status
|
581 |
|
582 | try {
|
583 | = [1, 2] ++ 3
|
584 | }
|
585 | echo List Int $_status
|
586 |
|
587 | try {
|
588 | = 3 ++ 'ab'
|
589 | }
|
590 | echo Int Str $_status
|
591 |
|
592 | ## STDOUT:
|
593 | string abcde
|
594 | list len=3
|
595 | ---
|
596 | Str Int 3
|
597 | List Int 3
|
598 | Int Str 3
|
599 | ## END
|
600 |
|
601 | #### s ~~ glob and s !~~ glob
|
602 | shopt -s oil:all
|
603 |
|
604 | if ('foo.py' ~~ '*.py') {
|
605 | echo yes
|
606 | }
|
607 | if ('foo.py' !~~ '*.sh') {
|
608 | echo no
|
609 | }
|
610 | ## STDOUT:
|
611 | yes
|
612 | no
|
613 | ## END
|
614 |
|
615 | #### Type Errors
|
616 | shopt --set parse_brace
|
617 |
|
618 | # TODO: It might be nice to get a message
|
619 | try {
|
620 | var x = {} + []
|
621 | }
|
622 | echo $_status
|
623 |
|
624 | try {
|
625 | setvar x = {} + 3
|
626 | }
|
627 | echo $_status
|
628 |
|
629 | try {
|
630 | = 'foo' ++ 3
|
631 | }
|
632 | echo $_status
|
633 |
|
634 | try {
|
635 | = 'foo' ++ 3
|
636 | }
|
637 | echo $_status
|
638 |
|
639 | ## STDOUT:
|
640 | 3
|
641 | 3
|
642 | 3
|
643 | 3
|
644 | ## END
|
645 |
|
646 |
|
647 | #### can't use ++ on integers
|
648 | var x = 12 ++ 3
|
649 | echo $x
|
650 | ## status: 3
|
651 | ## STDOUT:
|
652 | ## END
|
653 |
|
654 | #### can't do mystr ++ mylist
|
655 | = ["s"] + "t"
|
656 | ## status: 3
|
657 | ## STDOUT:
|
658 | ## END
|
659 |
|
660 |
|
661 | #### expression literals
|
662 | var e = ^[1 + 2]
|
663 |
|
664 | echo type=$[type(e)]
|
665 | echo eval=$[evalExpr(e)]
|
666 | ## STDOUT:
|
667 | type=Expr
|
668 | eval=3
|
669 | ## END
|
670 |
|
671 | #### expression literals, evaluation failure
|
672 | var e = ^[1 / 0]
|
673 | call evalExpr(e)
|
674 | ## status: 3
|
675 | ## STDOUT:
|
676 | ## END
|
677 |
|
678 | #### expression literals, lazy evaluation
|
679 | var x = 0
|
680 | var e = ^[x]
|
681 |
|
682 | setvar x = 1
|
683 | echo result=$[evalExpr(e)]
|
684 | ## STDOUT:
|
685 | result=1
|
686 | ## END
|
687 |
|
688 | #### expression literals, sugar for strings
|
689 | var x = 0
|
690 | var e = ^"x is $x"
|
691 |
|
692 | setvar x = 1
|
693 | echo result=$[evalExpr(e)]
|
694 | ## STDOUT:
|
695 | result=x is 1
|
696 | ## END
|