2018-12-31 00:20:07 -08:00
|
|
|
#!/usr/bin/env nomsu -V6.13.12.8
|
2018-05-15 18:55:55 -07:00
|
|
|
#
|
2018-01-11 18:51:21 -08:00
|
|
|
This file contains compile-time actions that define basic control flow structures
|
|
|
|
like "if" statements and loops.
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-02-02 15:48:28 -08:00
|
|
|
use "core/metaprogramming.nom"
|
|
|
|
use "core/operators.nom"
|
2018-01-11 18:51:21 -08:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# No-Op
|
2018-11-11 15:50:46 -08:00
|
|
|
test:
|
|
|
|
do nothing
|
2018-11-09 14:36:15 -08:00
|
|
|
(do nothing) compiles to ""
|
2017-09-21 00:10:26 -07:00
|
|
|
|
|
|
|
# Conditionals
|
2018-07-20 20:27:15 -07:00
|
|
|
test:
|
|
|
|
if (no):
|
|
|
|
barf "conditional fail"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
(if $condition $if_body) compiles to ("
|
2018-12-14 20:21:03 -08:00
|
|
|
if \($condition as lua expr) then
|
|
|
|
\($if_body as lua)
|
2018-12-30 19:04:34 -08:00
|
|
|
end
|
|
|
|
")
|
2018-07-20 20:27:15 -07:00
|
|
|
|
|
|
|
test:
|
|
|
|
unless (yes):
|
|
|
|
barf "conditional fail"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(unless $condition $unless_body) parses as (if (not $condition) $unless_body)
|
2018-12-30 23:58:47 -08:00
|
|
|
[
|
|
|
|
if $condition $if_body else $else_body, unless $condition $else_body else $if_body
|
|
|
|
] all compile to ("
|
2018-12-14 20:21:03 -08:00
|
|
|
if \($condition as lua expr) then
|
|
|
|
\($if_body as lua)
|
2018-11-09 14:36:15 -08:00
|
|
|
else
|
2018-12-14 20:21:03 -08:00
|
|
|
\($else_body as lua)
|
2018-12-30 19:04:34 -08:00
|
|
|
end
|
|
|
|
")
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Conditional expression (ternary operator)
|
2018-05-15 18:55:55 -07:00
|
|
|
# Note: this uses a function instead of "(condition and if_expr or else_expr)"
|
2018-12-31 00:55:58 -08:00
|
|
|
because that breaks if $if_expr is falsey, e.g. "x < 5 and false or 99"
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((1 if (yes) else 2) == 1)
|
|
|
|
assume ((1 if (no) else 2) == 2)
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
[
|
2018-12-14 20:21:03 -08:00
|
|
|
$when_true_expr if $condition else $when_false_expr
|
|
|
|
$when_true_expr if $condition otherwise $when_false_expr
|
|
|
|
$when_false_expr unless $condition else $when_true_expr
|
|
|
|
$when_false_expr unless $condition then $when_true_expr
|
2018-12-30 19:04:34 -08:00
|
|
|
] all compile to:
|
2018-12-31 00:55:58 -08:00
|
|
|
# If $when_true_expr is guaranteed to be truthy, we can use Lua's idiomatic
|
2018-06-18 15:44:29 -07:00
|
|
|
equivalent of a conditional expression: (cond and if_true or if_false)
|
2018-12-30 19:04:34 -08:00
|
|
|
if {.Text, .List, .Dict, .Number}.($when_true_expr.type):
|
|
|
|
return
|
2018-12-30 23:58:47 -08:00
|
|
|
Lua ("
|
|
|
|
(\($condition as lua expr) and \($when_true_expr as lua expr) or \
|
|
|
|
..\($when_false_expr as lua expr))
|
|
|
|
")
|
2018-07-17 23:08:13 -07:00
|
|
|
..else:
|
2018-06-18 15:44:29 -07:00
|
|
|
# Otherwise, need to do an anonymous inline function (yuck, too bad lua
|
|
|
|
doesn't have a proper ternary operator!)
|
|
|
|
To see why this is necessary consider: (random()<.5 and false or 99)
|
2018-12-30 19:04:34 -08:00
|
|
|
return
|
|
|
|
Lua ("
|
2018-12-14 20:21:03 -08:00
|
|
|
((function()
|
|
|
|
if \($condition as lua expr) then
|
|
|
|
return \($when_true_expr as lua expr)
|
2018-06-18 15:44:29 -07:00
|
|
|
else
|
2018-12-14 20:21:03 -08:00
|
|
|
return \($when_false_expr as lua expr)
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-12-30 19:04:34 -08:00
|
|
|
end)())
|
|
|
|
")
|
2018-01-23 19:22:20 -08:00
|
|
|
|
2018-09-26 13:05:28 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2017-09-21 00:10:26 -07:00
|
|
|
# GOTOs
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-12-14 20:21:03 -08:00
|
|
|
$i = 0
|
|
|
|
--- $loop ---
|
|
|
|
$i += 1
|
|
|
|
unless ($i == 10):
|
|
|
|
go to $loop
|
|
|
|
assume ($i == 10)
|
2018-11-19 17:44:46 -08:00
|
|
|
--- (Loop) ---
|
2018-12-14 20:21:03 -08:00
|
|
|
$i -= 1
|
|
|
|
unless ($i == 0):
|
2018-11-11 15:50:46 -08:00
|
|
|
go to (Loop)
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($i == 0)
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
(--- $label ---) compiles to ("
|
|
|
|
::label_\(
|
|
|
|
($label.stub, as lua id) if ($label.type == "Action") else
|
2018-12-14 20:21:03 -08:00
|
|
|
$label as lua identifier
|
2018-12-30 19:04:34 -08:00
|
|
|
)::
|
|
|
|
")
|
2018-11-09 14:36:15 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
(go to $label) compiles to ("
|
|
|
|
goto label_\(
|
|
|
|
($label.stub, as lua id) if ($label.type == "Action") else
|
2018-12-14 20:21:03 -08:00
|
|
|
$label as lua identifier
|
2018-12-30 19:04:34 -08:00
|
|
|
)
|
|
|
|
")
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-11 04:38:46 -08:00
|
|
|
# Basic loop control
|
2018-12-14 20:21:03 -08:00
|
|
|
(stop $var) compiles to:
|
|
|
|
if $var:
|
|
|
|
return (Lua "goto stop_\($var as lua identifier)")
|
2018-11-11 15:50:46 -08:00
|
|
|
..else:
|
|
|
|
return (Lua "break")
|
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(do next $var) compiles to:
|
|
|
|
if $var:
|
|
|
|
return (Lua "goto continue_\($var as lua identifier)")
|
2018-11-11 15:50:46 -08:00
|
|
|
..else:
|
|
|
|
return (Lua "goto continue")
|
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(---stop $var ---) compiles to "::stop_\($var as lua identifier)::"
|
|
|
|
(---next $var ---) compiles to "::continue_\($var as lua identifier)::"
|
2018-01-11 04:38:46 -08:00
|
|
|
|
2017-09-21 00:10:26 -07:00
|
|
|
# While loops
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-12-14 20:21:03 -08:00
|
|
|
$x = 0
|
|
|
|
repeat while ($x < 10): $x += 1
|
|
|
|
assume ($x == 10)
|
|
|
|
repeat while ($x < 20): stop
|
|
|
|
assume ($x == 10)
|
|
|
|
repeat while ($x < 20):
|
|
|
|
$x += 1
|
2018-11-11 15:50:46 -08:00
|
|
|
if (yes):
|
|
|
|
do next
|
2018-07-22 13:59:08 -07:00
|
|
|
barf "Failed to 'do next'"
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($x == 20)
|
2018-11-26 16:28:06 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(repeat while $condition $body) compiles to:
|
2018-12-30 23:58:47 -08:00
|
|
|
$lua =
|
|
|
|
Lua ("
|
|
|
|
while \($condition as lua expr) do
|
|
|
|
\($body as lua)
|
|
|
|
")
|
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n ::continue::"
|
2018-12-30 23:58:47 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\nend --while-loop"
|
2018-12-14 20:21:03 -08:00
|
|
|
return $lua
|
2018-11-26 16:28:06 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(repeat $body) parses as (repeat while (yes) $body)
|
|
|
|
(repeat until $condition $body) parses as (repeat while (not $condition) $body)
|
2018-01-23 19:22:20 -08:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2017-10-13 18:09:04 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-12-14 20:21:03 -08:00
|
|
|
$nums = []
|
|
|
|
for $x in 1 to 5:
|
2018-12-30 19:04:34 -08:00
|
|
|
$nums, add $x
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($nums == [1, 2, 3, 4, 5])
|
|
|
|
$nums = []
|
|
|
|
for $x in 1 to 5 via 2:
|
2018-12-30 19:04:34 -08:00
|
|
|
$nums, add $x
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($nums == [1, 3, 5])
|
|
|
|
$nums = []
|
|
|
|
for $outer in 1 to 100:
|
|
|
|
for $inner in $outer to ($outer + 2):
|
|
|
|
if ($inner == 2):
|
2018-12-30 19:04:34 -08:00
|
|
|
$nums, add -2
|
2018-12-14 20:21:03 -08:00
|
|
|
do next $inner
|
2018-12-30 19:04:34 -08:00
|
|
|
$nums, add $inner
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($inner == 5):
|
|
|
|
stop $outer
|
|
|
|
assume ($nums == [1, -2, 3, -2, 3, 4, 3, 4, 5])
|
2018-07-22 13:59:08 -07:00
|
|
|
|
2017-09-24 20:20:27 -07:00
|
|
|
# Numeric range for loops
|
2018-12-30 19:04:34 -08:00
|
|
|
[
|
2018-12-14 20:21:03 -08:00
|
|
|
for $var in $start to $stop by $step $body
|
|
|
|
for $var in $start to $stop via $step $body
|
2018-12-30 19:04:34 -08:00
|
|
|
] all compile to:
|
2018-06-18 15:44:29 -07:00
|
|
|
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua =
|
2018-12-30 23:58:47 -08:00
|
|
|
Lua ("
|
|
|
|
for \($var as lua identifier)=\($start as lua expr),\($stop as lua expr),\
|
|
|
|
..\($step as lua expr) do
|
|
|
|
")
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n " ($body as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n ::continue::"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next $var)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n " (\(---next $var ---) as lua)
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\nend -- numeric for " ($var as lua identifier) " loop"
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(stop $var)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua =
|
|
|
|
Lua ("
|
2018-12-14 20:21:03 -08:00
|
|
|
do -- scope for (stop \($var as lua identifier))
|
|
|
|
\$lua
|
|
|
|
\(\(---stop $var ---) as lua)
|
2018-12-30 19:04:34 -08:00
|
|
|
end -- scope for (stop \($var as lua identifier))
|
|
|
|
")
|
2018-12-14 20:21:03 -08:00
|
|
|
return $lua
|
2018-01-23 19:22:20 -08:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
(for $var in $start to $stop $body) parses as
|
2018-12-14 20:21:03 -08:00
|
|
|
for $var in $start to $stop via 1 $body
|
2018-01-08 18:53:57 -08:00
|
|
|
|
2018-11-26 16:21:42 -08:00
|
|
|
test:
|
2018-12-14 20:21:03 -08:00
|
|
|
$x = 0
|
2018-11-26 16:28:06 -08:00
|
|
|
repeat 5 times:
|
2018-12-14 20:21:03 -08:00
|
|
|
$x += 1
|
|
|
|
assume $x == 5
|
2018-11-26 16:21:42 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(repeat $n times $body) parses as (for (=lua "_XXX_") in 1 to $n $body)
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-12-14 20:21:03 -08:00
|
|
|
$a = [10, 20, 30, 40, 50]
|
|
|
|
$b = []
|
|
|
|
for $x in $a:
|
2018-12-30 19:04:34 -08:00
|
|
|
$b, add $x
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($a == $b)
|
|
|
|
$b = []
|
|
|
|
for $x in $a:
|
|
|
|
if ($x == 10):
|
|
|
|
do next $x
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($x == 50):
|
|
|
|
stop $x
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$b, add $x
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($b == [20, 30, 40])
|
2018-07-20 20:27:15 -07:00
|
|
|
|
2018-01-25 17:34:49 -08:00
|
|
|
# For-each loop (lua's "ipairs()")
|
2018-12-14 20:21:03 -08:00
|
|
|
(for $var in $iterable at $i $body) compiles to:
|
2018-09-21 00:30:28 -07:00
|
|
|
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua =
|
|
|
|
Lua ("
|
2018-12-14 20:21:03 -08:00
|
|
|
for \($i as lua identifier),\($var as lua identifier) in ipairs(\($iterable as lua expr)) do
|
2018-12-30 19:04:34 -08:00
|
|
|
\;
|
|
|
|
")
|
|
|
|
$lua, add ($body as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n ::continue::"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next $var)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n " (\(---next $var ---) as lua)
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\nend --for \($var as lua identifier) loop"
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(stop $var)):
|
|
|
|
$inner_lua = $lua
|
|
|
|
$lua = (Lua "do -- scope for stopping for-loop\n ")
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add $inner_lua "\n "
|
|
|
|
$lua, add (\(---stop $var ---) as lua)
|
|
|
|
$lua, add "\nend -- end of scope for stopping for-loop"
|
2018-12-14 20:21:03 -08:00
|
|
|
return $lua
|
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
(for $var in $iterable $body) parses as
|
2018-12-14 20:21:03 -08:00
|
|
|
for $var in $iterable at (=lua "__") $body
|
2018-11-26 16:21:42 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-12-30 19:04:34 -08:00
|
|
|
$d = {.a = 10, .b = 20, .c = 30, .d = 40, .e = 50}
|
2018-12-14 20:21:03 -08:00
|
|
|
$result = []
|
|
|
|
for $k = $v in $d:
|
|
|
|
if ($k == "a"):
|
|
|
|
do next $k
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($v == 20):
|
|
|
|
do next $v
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$result, add "\$k = \$v"
|
2018-12-14 20:21:03 -08:00
|
|
|
assume (($result sorted) == ["c = 30", "d = 40", "e = 50"])
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-01-03 00:52:01 -08:00
|
|
|
# Dict iteration (lua's "pairs()")
|
2018-12-30 19:04:34 -08:00
|
|
|
[for $key = $value in $iterable $body, for $key $value in $iterable $body]
|
2018-10-30 23:42:04 -07:00
|
|
|
..all compile to:
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua =
|
2018-12-30 23:58:47 -08:00
|
|
|
Lua ("
|
|
|
|
for \($key as lua identifier),\($value as lua identifier) in pairs(\
|
|
|
|
..\($iterable as lua expr)) do
|
|
|
|
")
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n " ($body as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n ::continue::"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next $key)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n " (\(---next $key ---) as lua)
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next $value)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n " (\(---next $value ---) as lua)
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\nend --foreach-loop"
|
2018-12-14 20:21:03 -08:00
|
|
|
$stop_labels = (Lua "")
|
|
|
|
if ($body has subtree \(stop $key)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$stop_labels, add "\n" (\(---stop $key ---) as lua)
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(stop $value)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$stop_labels, add "\n" (\(---stop $value ---) as lua)
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ((size of "\$stop_labels") > 0):
|
|
|
|
$inner_lua = $lua
|
2018-12-31 00:55:58 -08:00
|
|
|
$lua = (Lua "do -- scope for stopping for $ = $ loop\n ")
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add $inner_lua $stop_labels "\nend"
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
return $lua
|
2017-09-28 17:49:15 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-02 14:38:24 -07:00
|
|
|
when:
|
2018-07-22 15:01:05 -07:00
|
|
|
(1 == 2) (100 < 0):
|
|
|
|
barf "bad conditional"
|
2018-12-14 20:21:03 -08:00
|
|
|
(1 == 0) (1 == 1) $not_a_variable.x: do nothing
|
2018-07-22 15:01:05 -07:00
|
|
|
(1 == 1):
|
|
|
|
barf "bad conditional"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 15:01:05 -07:00
|
|
|
(1 == 2):
|
|
|
|
barf "bad conditional"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 15:01:05 -07:00
|
|
|
else:
|
|
|
|
barf "bad conditional"
|
2018-07-22 13:59:08 -07:00
|
|
|
|
2018-07-18 17:55:29 -07:00
|
|
|
# Multi-branch conditional (if..elseif..else)
|
2018-12-14 20:21:03 -08:00
|
|
|
(when $body) compiles to:
|
|
|
|
$code = (Lua "")
|
|
|
|
$clause = "if"
|
|
|
|
$else_allowed = (yes)
|
|
|
|
unless ($body.type is "Block"):
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $body "'if' expected a Block, but got a \($body.type)."
|
2018-12-30 23:58:47 -08:00
|
|
|
"Perhaps you forgot to put a ':' after 'if'?"
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
for $line in $body:
|
2018-12-30 19:04:34 -08:00
|
|
|
unless
|
|
|
|
(($line.type is "Action") and ((size of $line) >= 2)) and
|
2018-12-14 20:21:03 -08:00
|
|
|
$line.(size of $line) is "Block" syntax tree
|
2018-07-18 17:55:29 -07:00
|
|
|
..:
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $line "Invalid line for the body of an 'if' block." ("
|
|
|
|
Each line should contain one or more conditional expressions followed by a block, or "else"\
|
|
|
|
.. followed by a block.
|
|
|
|
")
|
2018-12-14 20:21:03 -08:00
|
|
|
$action = $line.(size of $line)
|
|
|
|
if (($line.1 is "else") and ((size of $line) == 2)):
|
|
|
|
unless $else_allowed:
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $line "You can't have two 'else' blocks."
|
2018-12-30 23:58:47 -08:00
|
|
|
"Merge all of the 'else' blocks together."
|
2018-10-30 23:42:04 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
unless ((size of "\$code") > 0):
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $line
|
|
|
|
.."You can't have an 'else' block without a preceding condition" ("
|
|
|
|
If you want the code in this block to always execute, you don't need a conditional block \
|
|
|
|
..around it. Otherwise, make sure the 'else' block comes last.
|
|
|
|
")
|
|
|
|
|
2018-12-30 23:58:47 -08:00
|
|
|
$code, add "\nelse\n " ($action as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
$else_allowed = (no)
|
2018-07-18 17:55:29 -07:00
|
|
|
..else:
|
2018-12-30 19:04:34 -08:00
|
|
|
$code, add $clause " "
|
2018-12-14 20:21:03 -08:00
|
|
|
for $i in 1 to ((size of $line) - 1):
|
|
|
|
if ($i > 1):
|
2018-12-30 19:04:34 -08:00
|
|
|
$code, add " or "
|
|
|
|
$code, add ($line.$i as lua expr)
|
2018-12-30 23:58:47 -08:00
|
|
|
$code, add " then\n " ($action as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
$clause = "\nelseif"
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ((size of "\$code") == 0):
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $body "'if' block has an empty body."
|
2018-12-30 23:58:47 -08:00
|
|
|
"This means nothing would happen, so the 'if' block should be deleted."
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$code, add "\nend --when"
|
2018-12-14 20:21:03 -08:00
|
|
|
return $code
|
2017-12-04 17:35:47 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
if 5 is:
|
2018-07-22 15:01:05 -07:00
|
|
|
1 2 3:
|
|
|
|
barf "bad switch statement"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
|
|
|
4 5:
|
|
|
|
do nothing
|
|
|
|
|
2018-07-22 15:01:05 -07:00
|
|
|
5 6:
|
|
|
|
barf "bad switch statement"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 15:01:05 -07:00
|
|
|
else:
|
|
|
|
barf "bad switch statement"
|
2018-07-20 20:27:15 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
# Switch statement
|
2018-12-14 20:21:03 -08:00
|
|
|
[if $branch_value is $body, when $branch_value is $body] all compile to:
|
|
|
|
$code = (Lua "")
|
|
|
|
$clause = "if"
|
|
|
|
$else_allowed = (yes)
|
2018-09-06 12:46:39 -07:00
|
|
|
define mangler
|
2018-12-14 20:21:03 -08:00
|
|
|
unless ($body.type is "Block"):
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $body "'if' expected a Block, but got a \($body.type)"
|
2018-12-30 23:58:47 -08:00
|
|
|
"Perhaps you forgot to put a ':' after the 'is'?"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
for $line in $body:
|
2018-12-30 19:04:34 -08:00
|
|
|
unless
|
|
|
|
(($line.type is "Action") and ((size of $line) >= 2)) and
|
2018-12-14 20:21:03 -08:00
|
|
|
$line.(size of $line) is "Block" syntax tree
|
2018-07-18 17:55:29 -07:00
|
|
|
..:
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $line "Invalid line for 'if' block." ("
|
|
|
|
Each line should contain expressions followed by a block, or "else" followed by a block
|
|
|
|
")
|
2018-12-14 20:21:03 -08:00
|
|
|
$action = $line.(size of $line)
|
|
|
|
if (($line.1 is "else") and ((size of $line) == 2)):
|
|
|
|
unless $else_allowed:
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $line "You can't have two 'else' blocks."
|
2018-12-30 23:58:47 -08:00
|
|
|
"Merge all of the 'else' blocks together."
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
unless ((size of "\$code") > 0):
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $line
|
|
|
|
.."You can't have an 'else' block without a preceding condition" ("
|
|
|
|
If you want the code in this block to always execute, you don't need a conditional block \
|
|
|
|
..around it. Otherwise, make sure the 'else' block comes last.
|
|
|
|
")
|
|
|
|
|
2018-12-30 23:58:47 -08:00
|
|
|
$code, add "\nelse\n " ($action as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
$else_allowed = (no)
|
2018-07-18 17:55:29 -07:00
|
|
|
..else:
|
2018-12-30 19:04:34 -08:00
|
|
|
$code, add $clause " "
|
2018-12-14 20:21:03 -08:00
|
|
|
for $i in 1 to ((size of $line) - 1):
|
|
|
|
if ($i > 1):
|
2018-12-30 19:04:34 -08:00
|
|
|
$code, add " or "
|
|
|
|
$code, add "\(mangle "branch value") == " ($line.$i as lua expr)
|
2018-12-30 23:58:47 -08:00
|
|
|
$code, add " then\n " ($action as lua)
|
2018-12-14 20:21:03 -08:00
|
|
|
$clause = "\nelseif"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ((size of "\$code") == 0):
|
2018-12-30 19:04:34 -08:00
|
|
|
compile error at $body "'if' block has an empty body."
|
2018-12-30 23:58:47 -08:00
|
|
|
"This means nothing would happen, so the 'if' block should be deleted."
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$code, add "\nend --when"
|
|
|
|
return
|
|
|
|
Lua ("
|
2018-12-31 00:55:58 -08:00
|
|
|
do --if $ is...
|
2018-12-14 20:21:03 -08:00
|
|
|
local \(mangle "branch value") = \($branch_value as lua expr)
|
|
|
|
\$code
|
2018-12-31 00:55:58 -08:00
|
|
|
end -- if $ is...
|
2018-12-30 19:04:34 -08:00
|
|
|
")
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
# Do/finally
|
2018-12-30 19:04:34 -08:00
|
|
|
(do $action) compiles to ("
|
|
|
|
do
|
|
|
|
\($action as lua)
|
|
|
|
end -- do
|
|
|
|
")
|
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-11 15:50:46 -08:00
|
|
|
assume ((result of: return 99) == 99)
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-05-10 22:47:03 -07:00
|
|
|
# Inline thunk:
|
2018-12-14 20:21:03 -08:00
|
|
|
(result of $body) compiles to "\(\(-> $body) as lua)()"
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-12-14 20:21:03 -08:00
|
|
|
$t = [1, [2, [[3], 4], 5, [[[6]]]]]
|
|
|
|
$flat = []
|
|
|
|
for $ in recursive $t:
|
|
|
|
if ((lua type of $) is "table"):
|
|
|
|
for $2 in $:
|
|
|
|
recurse $ on $2
|
2018-11-11 15:50:46 -08:00
|
|
|
..else:
|
2018-12-30 19:04:34 -08:00
|
|
|
$flat, add $
|
2018-12-14 20:21:03 -08:00
|
|
|
assume (sorted $flat) == [1, 2, 3, 4, 5, 6]
|
2018-07-22 13:59:08 -07:00
|
|
|
|
2018-06-21 19:12:59 -07:00
|
|
|
# Recurion control flow
|
2018-12-14 20:21:03 -08:00
|
|
|
(for $var in recursive $structure $body) compiles to:
|
2018-07-17 23:08:13 -07:00
|
|
|
with local compile actions:
|
2018-09-06 12:46:39 -07:00
|
|
|
define mangler
|
2018-12-30 19:04:34 -08:00
|
|
|
(recurse $v on $x) compiles to
|
2018-12-14 20:21:03 -08:00
|
|
|
Lua "table.insert(\(mangle "stack \($v.1)"), \($x as lua expr))"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua =
|
|
|
|
Lua ("
|
2018-12-14 20:21:03 -08:00
|
|
|
do
|
|
|
|
local \(mangle "stack \($var.1)") = List{\($structure as lua expr)}
|
|
|
|
while #\(mangle "stack \($var.1)") > 0 do
|
|
|
|
\($var as lua expr) = table.remove(\(mangle "stack \($var.1)"), 1)
|
2018-12-30 19:04:34 -08:00
|
|
|
\($body as lua)
|
|
|
|
")
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n ::continue::"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(do next $var)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n \(\(---next $var ---) as lua)"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n end -- Recursive loop"
|
2018-12-14 20:21:03 -08:00
|
|
|
if ($body has subtree \(stop $var)):
|
2018-12-30 19:04:34 -08:00
|
|
|
$lua, add "\n \(\(---stop $var ---) as lua)"
|
|
|
|
$lua, add "\nend -- Recursive scope"
|
2018-12-14 20:21:03 -08:00
|
|
|
return $lua
|