Misc changes, including text indented interpolations are now indented
relative to the text, not the opening '("', code objects can now remove all free vars, the REPL uses global vars. Error API is changing a bit.
This commit is contained in:
parent
0760d4fb64
commit
b6d3cbd61c
51
code_obj.lua
51
code_obj.lua
@ -320,6 +320,10 @@ do
|
|||||||
return self:dirty()
|
return self:dirty()
|
||||||
end,
|
end,
|
||||||
remove_free_vars = function(self, vars)
|
remove_free_vars = function(self, vars)
|
||||||
|
if vars == nil then
|
||||||
|
vars = nil
|
||||||
|
end
|
||||||
|
vars = vars or self:get_free_vars()
|
||||||
if not (#vars > 0) then
|
if not (#vars > 0) then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
@ -351,33 +355,34 @@ do
|
|||||||
end
|
end
|
||||||
return self:dirty()
|
return self:dirty()
|
||||||
end,
|
end,
|
||||||
|
get_free_vars = function(self)
|
||||||
|
local vars, seen = { }, { }
|
||||||
|
local gather_from
|
||||||
|
gather_from = function(self)
|
||||||
|
local _list_0 = self.free_vars
|
||||||
|
for _index_0 = 1, #_list_0 do
|
||||||
|
local var = _list_0[_index_0]
|
||||||
|
if not (seen[var]) then
|
||||||
|
seen[var] = true
|
||||||
|
vars[#vars + 1] = var
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local _list_1 = self.bits
|
||||||
|
for _index_0 = 1, #_list_1 do
|
||||||
|
local bit = _list_1[_index_0]
|
||||||
|
if not (type(bit) == 'string') then
|
||||||
|
gather_from(bit)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
gather_from(self)
|
||||||
|
return vars
|
||||||
|
end,
|
||||||
declare_locals = function(self, to_declare)
|
declare_locals = function(self, to_declare)
|
||||||
if to_declare == nil then
|
if to_declare == nil then
|
||||||
to_declare = nil
|
to_declare = nil
|
||||||
end
|
end
|
||||||
if to_declare == nil then
|
to_declare = to_declare or self:get_free_vars()
|
||||||
local seen
|
|
||||||
to_declare, seen = { }, { }
|
|
||||||
local gather_from
|
|
||||||
gather_from = function(self)
|
|
||||||
local _list_0 = self.free_vars
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local var = _list_0[_index_0]
|
|
||||||
if not (seen[var]) then
|
|
||||||
seen[var] = true
|
|
||||||
to_declare[#to_declare + 1] = var
|
|
||||||
end
|
|
||||||
end
|
|
||||||
local _list_1 = self.bits
|
|
||||||
for _index_0 = 1, #_list_1 do
|
|
||||||
local bit = _list_1[_index_0]
|
|
||||||
if not (type(bit) == 'string') then
|
|
||||||
gather_from(bit)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
gather_from(self)
|
|
||||||
end
|
|
||||||
if #to_declare > 0 then
|
if #to_declare > 0 then
|
||||||
self:remove_free_vars(to_declare)
|
self:remove_free_vars(to_declare)
|
||||||
self:prepend("local " .. tostring(concat(to_declare, ", ")) .. ";\n")
|
self:prepend("local " .. tostring(concat(to_declare, ", ")) .. ";\n")
|
||||||
|
@ -179,7 +179,8 @@ class LuaCode extends Code
|
|||||||
seen[var] = true
|
seen[var] = true
|
||||||
@dirty!
|
@dirty!
|
||||||
|
|
||||||
remove_free_vars: (vars)=>
|
remove_free_vars: (vars=nil)=>
|
||||||
|
vars or= @get_free_vars!
|
||||||
return unless #vars > 0
|
return unless #vars > 0
|
||||||
removals = {}
|
removals = {}
|
||||||
for var in *vars
|
for var in *vars
|
||||||
@ -198,18 +199,21 @@ class LuaCode extends Code
|
|||||||
stack[#stack+1] = b
|
stack[#stack+1] = b
|
||||||
@dirty!
|
@dirty!
|
||||||
|
|
||||||
|
get_free_vars: =>
|
||||||
|
vars, seen = {}, {}
|
||||||
|
gather_from = =>
|
||||||
|
for var in *@free_vars
|
||||||
|
unless seen[var]
|
||||||
|
seen[var] = true
|
||||||
|
vars[#vars+1] = var
|
||||||
|
for bit in *@bits
|
||||||
|
unless type(bit) == 'string'
|
||||||
|
gather_from bit
|
||||||
|
gather_from self
|
||||||
|
return vars
|
||||||
|
|
||||||
declare_locals: (to_declare=nil)=>
|
declare_locals: (to_declare=nil)=>
|
||||||
if to_declare == nil
|
to_declare or= @get_free_vars!
|
||||||
to_declare, seen = {}, {}
|
|
||||||
gather_from = =>
|
|
||||||
for var in *@free_vars
|
|
||||||
unless seen[var]
|
|
||||||
seen[var] = true
|
|
||||||
to_declare[#to_declare+1] = var
|
|
||||||
for bit in *@bits
|
|
||||||
unless type(bit) == 'string'
|
|
||||||
gather_from bit
|
|
||||||
gather_from self
|
|
||||||
if #to_declare > 0
|
if #to_declare > 0
|
||||||
@remove_free_vars to_declare
|
@remove_free_vars to_declare
|
||||||
@prepend "local #{concat to_declare, ", "};\n"
|
@prepend "local #{concat to_declare, ", "};\n"
|
||||||
|
@ -251,18 +251,6 @@ List = function(t)
|
|||||||
return error("Unsupported List type: " .. type(t))
|
return error("Unsupported List type: " .. type(t))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
local walk_items
|
|
||||||
walk_items = function(self, i)
|
|
||||||
i = i + 1
|
|
||||||
local k, v = next(self.table, self.key)
|
|
||||||
if k ~= nil then
|
|
||||||
self.key = k
|
|
||||||
return i, Dict({
|
|
||||||
key = k,
|
|
||||||
value = v
|
|
||||||
})
|
|
||||||
end
|
|
||||||
end
|
|
||||||
local _dict_mt = {
|
local _dict_mt = {
|
||||||
__type = "Dict",
|
__type = "Dict",
|
||||||
__eq = function(self, other)
|
__eq = function(self, other)
|
||||||
@ -321,12 +309,6 @@ local _dict_mt = {
|
|||||||
return _accum_0
|
return _accum_0
|
||||||
end)(), ", ") .. "}"
|
end)(), ", ") .. "}"
|
||||||
end,
|
end,
|
||||||
__ipairs = function(self)
|
|
||||||
return walk_items, {
|
|
||||||
table = self,
|
|
||||||
key = nil
|
|
||||||
}, 0
|
|
||||||
end,
|
|
||||||
__band = function(self, other)
|
__band = function(self, other)
|
||||||
return Dict((function()
|
return Dict((function()
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
@ -339,30 +321,28 @@ local _dict_mt = {
|
|||||||
end)())
|
end)())
|
||||||
end,
|
end,
|
||||||
__bor = function(self, other)
|
__bor = function(self, other)
|
||||||
local ret
|
local ret = Dict((function()
|
||||||
do
|
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
_tbl_0[k] = v
|
_tbl_0[k] = v
|
||||||
end
|
end
|
||||||
ret = _tbl_0
|
return _tbl_0
|
||||||
end
|
end)())
|
||||||
for k, v in pairs(other) do
|
for k, v in pairs(other) do
|
||||||
if ret[k] == nil then
|
if ret[k] == nil then
|
||||||
ret[k] = v
|
ret[k] = v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return Dict(ret)
|
return ret
|
||||||
end,
|
end,
|
||||||
__bxor = function(self, other)
|
__bxor = function(self, other)
|
||||||
local ret
|
local ret = Dict((function()
|
||||||
do
|
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
_tbl_0[k] = v
|
_tbl_0[k] = v
|
||||||
end
|
end
|
||||||
ret = _tbl_0
|
return _tbl_0
|
||||||
end
|
end)())
|
||||||
for k, v in pairs(other) do
|
for k, v in pairs(other) do
|
||||||
if ret[k] == nil then
|
if ret[k] == nil then
|
||||||
ret[k] = v
|
ret[k] = v
|
||||||
@ -370,17 +350,16 @@ local _dict_mt = {
|
|||||||
ret[k] = nil
|
ret[k] = nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return Dict(ret)
|
return ret
|
||||||
end,
|
end,
|
||||||
__add = function(self, other)
|
__add = function(self, other)
|
||||||
local ret
|
local ret = Dict((function()
|
||||||
do
|
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
_tbl_0[k] = v
|
_tbl_0[k] = v
|
||||||
end
|
end
|
||||||
ret = _tbl_0
|
return _tbl_0
|
||||||
end
|
end)())
|
||||||
for k, v in pairs(other) do
|
for k, v in pairs(other) do
|
||||||
if ret[k] == nil then
|
if ret[k] == nil then
|
||||||
ret[k] = v
|
ret[k] = v
|
||||||
@ -388,17 +367,16 @@ local _dict_mt = {
|
|||||||
ret[k] = ret[k] + v
|
ret[k] = ret[k] + v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return Dict(ret)
|
return ret
|
||||||
end,
|
end,
|
||||||
__sub = function(self, other)
|
__sub = function(self, other)
|
||||||
local ret
|
local ret = Dict((function()
|
||||||
do
|
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
_tbl_0[k] = v
|
_tbl_0[k] = v
|
||||||
end
|
end
|
||||||
ret = _tbl_0
|
return _tbl_0
|
||||||
end
|
end)())
|
||||||
for k, v in pairs(other) do
|
for k, v in pairs(other) do
|
||||||
if ret[k] == nil then
|
if ret[k] == nil then
|
||||||
ret[k] = -v
|
ret[k] = -v
|
||||||
@ -406,7 +384,7 @@ local _dict_mt = {
|
|||||||
ret[k] = ret[k] - v
|
ret[k] = ret[k] - v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return Dict(ret)
|
return ret
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
Dict = function(t)
|
Dict = function(t)
|
||||||
@ -430,11 +408,6 @@ Dict = function(t)
|
|||||||
return error("Unsupported Dict type: " .. type(t))
|
return error("Unsupported Dict type: " .. type(t))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
for i, entry in ipairs(Dict({
|
|
||||||
x = 99
|
|
||||||
})) do
|
|
||||||
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
|
||||||
end
|
|
||||||
do
|
do
|
||||||
local reverse, upper, lower, find, byte, match, gmatch, gsub, sub, format, rep
|
local reverse, upper, lower, find, byte, match, gmatch, gsub, sub, format, rep
|
||||||
do
|
do
|
||||||
@ -502,9 +475,9 @@ do
|
|||||||
return (match(self, patt))
|
return (match(self, patt))
|
||||||
end,
|
end,
|
||||||
matching_groups = function(self, patt)
|
matching_groups = function(self, patt)
|
||||||
return {
|
return List({
|
||||||
match(self, patt)
|
match(self, patt)
|
||||||
}
|
})
|
||||||
end,
|
end,
|
||||||
[as_lua_id("* 1")] = function(self, n)
|
[as_lua_id("* 1")] = function(self, n)
|
||||||
return rep(self, n)
|
return rep(self, n)
|
||||||
|
@ -106,13 +106,6 @@ List = (t)->
|
|||||||
return l
|
return l
|
||||||
else error("Unsupported List type: "..type(t))
|
else error("Unsupported List type: "..type(t))
|
||||||
|
|
||||||
walk_items = (i)=>
|
|
||||||
i = i + 1
|
|
||||||
k, v = next(@table, @key)
|
|
||||||
if k != nil
|
|
||||||
@key = k
|
|
||||||
return i, Dict{key:k, value:v}
|
|
||||||
|
|
||||||
_dict_mt =
|
_dict_mt =
|
||||||
__type: "Dict"
|
__type: "Dict"
|
||||||
__eq: (other)=>
|
__eq: (other)=>
|
||||||
@ -133,32 +126,31 @@ _dict_mt =
|
|||||||
"{"..concat(["#{as_nomsu(k)}: #{as_nomsu(v)}" for k,v in pairs @], ", ").."}"
|
"{"..concat(["#{as_nomsu(k)}: #{as_nomsu(v)}" for k,v in pairs @], ", ").."}"
|
||||||
as_lua: =>
|
as_lua: =>
|
||||||
"Dict{"..concat(["[ #{as_lua(k)}]= #{as_lua(v)}" for k,v in pairs @], ", ").."}"
|
"Dict{"..concat(["[ #{as_lua(k)}]= #{as_lua(v)}" for k,v in pairs @], ", ").."}"
|
||||||
__ipairs: => walk_items, {table:@, key:nil}, 0
|
|
||||||
__band: (other)=>
|
__band: (other)=>
|
||||||
Dict{k,v for k,v in pairs(@) when other[k] != nil}
|
Dict{k,v for k,v in pairs(@) when other[k] != nil}
|
||||||
__bor: (other)=>
|
__bor: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
return Dict(ret)
|
return ret
|
||||||
__bxor: (other)=>
|
__bxor: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
else ret[k] = nil
|
else ret[k] = nil
|
||||||
return Dict(ret)
|
return ret
|
||||||
__add: (other)=>
|
__add: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
else ret[k] += v
|
else ret[k] += v
|
||||||
return Dict(ret)
|
return ret
|
||||||
__sub: (other)=>
|
__sub: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = -v
|
if ret[k] == nil then ret[k] = -v
|
||||||
else ret[k] -= v
|
else ret[k] -= v
|
||||||
return Dict(ret)
|
return ret
|
||||||
Dict = (t)->
|
Dict = (t)->
|
||||||
if type(t) == 'table'
|
if type(t) == 'table'
|
||||||
return setmetatable(t, _dict_mt)
|
return setmetatable(t, _dict_mt)
|
||||||
@ -171,8 +163,6 @@ Dict = (t)->
|
|||||||
return d
|
return d
|
||||||
else error("Unsupported Dict type: "..type(t))
|
else error("Unsupported Dict type: "..type(t))
|
||||||
|
|
||||||
for i,entry in ipairs(Dict({x:99}))
|
|
||||||
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
|
||||||
|
|
||||||
do
|
do
|
||||||
{:reverse, :upper, :lower, :find, :byte, :match, :gmatch, :gsub, :sub, :format, :rep} = string
|
{:reverse, :upper, :lower, :find, :byte, :match, :gmatch, :gsub, :sub, :format, :rep} = string
|
||||||
@ -204,7 +194,7 @@ do
|
|||||||
line_position_at: (i)=> select(3, line_at(@, i))
|
line_position_at: (i)=> select(3, line_at(@, i))
|
||||||
matches: (patt)=> match(@, patt) and true or false
|
matches: (patt)=> match(@, patt) and true or false
|
||||||
matching: (patt)=> (match(@, patt))
|
matching: (patt)=> (match(@, patt))
|
||||||
matching_groups: (patt)=> {match(@, patt)}
|
matching_groups: (patt)=> List{match(@, patt)}
|
||||||
[as_lua_id "* 1"]: (n)=> rep(@, n)
|
[as_lua_id "* 1"]: (n)=> rep(@, n)
|
||||||
all_matches_of: (patt)=>
|
all_matches_of: (patt)=>
|
||||||
result = {}
|
result = {}
|
||||||
|
@ -37,7 +37,6 @@ test:
|
|||||||
test:
|
test:
|
||||||
$dict = {.x = 1, .y = 2, .z = 3}
|
$dict = {.x = 1, .y = 2, .z = 3}
|
||||||
assume (size of $dict) == 3
|
assume (size of $dict) == 3
|
||||||
assume [: for $ in {.x = 1}: add $] == [{.key = "x", .value = 1}]
|
|
||||||
assume [: for $k = $v in {.x = 1}: add {.key = $k, .value = $v}] ==
|
assume [: for $k = $v in {.x = 1}: add {.key = $k, .value = $v}] ==
|
||||||
[{.key = "x", .value = 1}]
|
[{.key = "x", .value = 1}]
|
||||||
assume ({.x = 1, .y = 1} + {.y = 10, .z = 10}) == {.x = 1, .y = 11, .z = 10}
|
assume ({.x = 1, .y = 1} + {.y = 10, .z = 10}) == {.x = 1, .y = 11, .z = 10}
|
||||||
|
@ -5,7 +5,6 @@
|
|||||||
|
|
||||||
use "core/metaprogramming.nom"
|
use "core/metaprogramming.nom"
|
||||||
use "core/operators.nom"
|
use "core/operators.nom"
|
||||||
use "core/errors.nom"
|
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
@ -448,32 +447,6 @@ test:
|
|||||||
end -- do
|
end -- do
|
||||||
")
|
")
|
||||||
|
|
||||||
test:
|
|
||||||
$d = {}
|
|
||||||
try:
|
|
||||||
do:
|
|
||||||
$d.x = "bad"
|
|
||||||
barf
|
|
||||||
..then always:
|
|
||||||
$d.x = "good"
|
|
||||||
assume ($d.x == "good")
|
|
||||||
|
|
||||||
(do $action then always $final_action) compiles to:
|
|
||||||
define mangler
|
|
||||||
return
|
|
||||||
Lua ("
|
|
||||||
do
|
|
||||||
local \(mangle "fell_through") = false
|
|
||||||
local \(mangle "ok"), \(mangle "ret") = pcall(function()
|
|
||||||
\($action as lua)
|
|
||||||
\(mangle "fell_through") = true
|
|
||||||
end)
|
|
||||||
\($final_action as lua)
|
|
||||||
if not \(mangle "ok") then error(ret, 0) end
|
|
||||||
if not \(mangle "fell_through") then return ret end
|
|
||||||
end
|
|
||||||
")
|
|
||||||
|
|
||||||
test:
|
test:
|
||||||
assume ((result of: return 99) == 99)
|
assume ((result of: return 99) == 99)
|
||||||
|
|
||||||
|
121
core/errors.nom
121
core/errors.nom
@ -3,11 +3,12 @@
|
|||||||
This file contains basic error reporting code
|
This file contains basic error reporting code
|
||||||
|
|
||||||
use "core/metaprogramming.nom"
|
use "core/metaprogramming.nom"
|
||||||
|
use "core/operators.nom"
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
(barf $msg) compiles to
|
(fail $msg) compiles to
|
||||||
"error(\(=lua "\$msg and \($msg as lua expr) or 'nil'"), 0);"
|
"error(\(($msg as lua expr) if $msg else "nil"), 0);"
|
||||||
|
|
||||||
(assume $condition) compiles to:
|
(assume $condition) compiles to:
|
||||||
lua> ("
|
lua> ("
|
||||||
@ -37,62 +38,86 @@ use "core/metaprogramming.nom"
|
|||||||
end
|
end
|
||||||
")
|
")
|
||||||
|
|
||||||
(assume $condition or barf $message) compiles to ("
|
|
||||||
if not \($condition as lua expr) then
|
|
||||||
error(\($message as lua expr), 0)
|
|
||||||
end
|
|
||||||
")
|
|
||||||
|
|
||||||
test:
|
test:
|
||||||
try (barf) and if it succeeds:
|
try: fail
|
||||||
barf "try failed."
|
|
||||||
$worked = (no)
|
$worked = (no)
|
||||||
try (barf) and if it barfs:
|
try: fail
|
||||||
|
..if it fails:
|
||||||
$worked = (yes)
|
$worked = (yes)
|
||||||
assume $worked or barf "try/catch failed"
|
..if it succeeds:
|
||||||
$x = 1
|
fail "'try' incorrectly ran success case."
|
||||||
try:
|
|
||||||
$x = 2
|
|
||||||
do (barf) then always: $x = 3
|
|
||||||
..and if it barfs:
|
|
||||||
do nothing
|
|
||||||
assume ($x == 3) or barf "do/then always failed"
|
|
||||||
|
|
||||||
|
unless $worked:
|
||||||
|
fail "'try' failed to recover from failure"
|
||||||
# Try/except
|
# Try/except
|
||||||
[
|
[
|
||||||
try $action and if it succeeds $success or if it barfs $msg $fallback
|
try $action if it succeeds $success if it fails $fallback
|
||||||
try $action and if it barfs $msg $fallback or if it succeeds $success
|
try $action if it fails $fallback if it succeeds $success
|
||||||
] all compile to ("
|
] all compile to:
|
||||||
do
|
$success_lua = ($success as lua)
|
||||||
local fell_through = false
|
if ((#"\$success_lua") > 0): $success_lua, add "\n"
|
||||||
local err, erred = nil, false
|
$success_lua, prepend "-- Success:\n"
|
||||||
local ok, ret = xpcall(function()
|
$success_lua, add "if not _fell_through then return table.unpack(_result, 2) end"
|
||||||
\($action as lua)
|
$fallback_lua = ($fallback as lua)
|
||||||
fell_through = true
|
if ((#"\$fallback_lua") > 0):
|
||||||
end, function(\(=lua "\$fallback and \($msg as lua expr) or ''"))
|
$fallback_lua, prepend "\nlocal function failure() return _result[2] end\n"
|
||||||
local ok, ret = pcall(function()
|
$fallback_lua, prepend "-- Failure:"
|
||||||
\((=lua "\$fallback or \$msg") as lua)
|
return
|
||||||
end)
|
Lua ("
|
||||||
if not ok then err, erred = ret, true end
|
do
|
||||||
end)
|
local _fell_through = false
|
||||||
if ok then
|
local _result = {pcall(function()
|
||||||
\($success as lua)
|
\($action as lua)
|
||||||
if not fell_through then
|
_fell_through = true
|
||||||
return ret
|
end)}
|
||||||
|
if _result[1] then
|
||||||
|
\$success_lua
|
||||||
|
else
|
||||||
|
\$fallback_lua
|
||||||
|
end
|
||||||
end
|
end
|
||||||
elseif erred then
|
")
|
||||||
error(err, 0)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
")
|
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
(try $action) parses as
|
(try $action) parses as
|
||||||
try $action and if it succeeds (do nothing) or if it barfs (do nothing)
|
try $action if it succeeds (do nothing) if it fails (do nothing)
|
||||||
|
|
||||||
(try $action and if it barfs $msg $fallback) parses as
|
(try $action if it fails $msg $fallback) parses as
|
||||||
try $action and if it succeeds (do nothing) or if it barfs $msg $fallback
|
try $action if it succeeds (do nothing) if it fails $msg $fallback
|
||||||
|
|
||||||
(try $action and if it succeeds $success) parses as
|
(try $action if it succeeds $success) parses as
|
||||||
try $action and if it succeeds $success or if it barfs (do nothing)
|
try $action if it succeeds $success if it fails (do nothing)
|
||||||
|
|
||||||
|
(try $action if it fails $fallback if it succeeds $success) parses as
|
||||||
|
try $action if it succeeds $success if it fails $fallback
|
||||||
|
|
||||||
|
test:
|
||||||
|
$success = (no)
|
||||||
|
try:
|
||||||
|
do: fail
|
||||||
|
..then always:
|
||||||
|
$success = (yes)
|
||||||
|
..if it succeeds:
|
||||||
|
fail "'try ... then always ...' didn't propagate failure"
|
||||||
|
|
||||||
|
unless $success:
|
||||||
|
fail "'try ... then always ...' didn't execute the 'always' code"
|
||||||
|
|
||||||
|
(do $action then always $final_action) compiles to ("
|
||||||
|
do -- do/then always
|
||||||
|
local _fell_through = false
|
||||||
|
local _results = {pcall(function()
|
||||||
|
\($action as lua)
|
||||||
|
_fell_through = true
|
||||||
|
end)}
|
||||||
|
\($final_action as lua)
|
||||||
|
if not _results[1] then error(_results[2], 0) end
|
||||||
|
if not _fell_through then return table.unpack(_results, 2) end
|
||||||
|
end
|
||||||
|
")
|
||||||
|
|
||||||
|
~~~
|
||||||
|
|
||||||
|
(barf $) parses as (fail $)
|
||||||
|
(assume $1 or barf $2) parses as (unless $1: fail $2)
|
||||||
|
@ -308,11 +308,11 @@ externally ($ is $kind syntax tree) means
|
|||||||
($tree with $t -> $replacement) compiles to ("
|
($tree with $t -> $replacement) compiles to ("
|
||||||
\($tree as lua expr):map(function(\($t as lua expr))
|
\($tree as lua expr):map(function(\($t as lua expr))
|
||||||
\(
|
\(
|
||||||
=lua ("
|
=lua ("
|
||||||
\$replacement.type == 'Block' and \($replacement as lua) or 'return '..\
|
\$replacement.type == 'Block' and \($replacement as lua) or 'return '..\
|
||||||
..\($replacement as lua expr)
|
..\($replacement as lua expr)
|
||||||
")
|
")
|
||||||
)
|
)
|
||||||
end)
|
end)
|
||||||
")
|
")
|
||||||
|
|
||||||
|
@ -3,7 +3,6 @@
|
|||||||
This file contains definitions of operators like "+" and "and".
|
This file contains definitions of operators like "+" and "and".
|
||||||
|
|
||||||
use "core/metaprogramming.nom"
|
use "core/metaprogramming.nom"
|
||||||
use "core/errors.nom"
|
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
@ -168,7 +168,11 @@ blank_text_lines <-
|
|||||||
{~ (%nl ((ws* -> '') (&%nl / !.) / (=curr_indent -> '') &[^%nl]))+ ~}
|
{~ (%nl ((ws* -> '') (&%nl / !.) / (=curr_indent -> '') &[^%nl]))+ ~}
|
||||||
|
|
||||||
text_interpolation <-
|
text_interpolation <-
|
||||||
("\" (indented_block (blank_lines =curr_indent "..")? / indented_expression))
|
({|
|
||||||
|
-- %indentation will backtrack and match the actual indentation of the current line
|
||||||
|
"\" {:curr_indent: %indentation :}
|
||||||
|
(indented_block (blank_lines =curr_indent "..")? / indented_expression)
|
||||||
|
|} -> unpack)
|
||||||
/ inline_text_interpolation
|
/ inline_text_interpolation
|
||||||
|
|
||||||
|
|
||||||
|
@ -304,7 +304,7 @@ local compile = setmetatable({
|
|||||||
lua:add(",")
|
lua:add(",")
|
||||||
end
|
end
|
||||||
if lua:trailing_line_len() + #(entry_lua:text():match("^[\n]*")) > MAX_LINE then
|
if lua:trailing_line_len() + #(entry_lua:text():match("^[\n]*")) > MAX_LINE then
|
||||||
lua:add("\n")
|
lua:add("\n ")
|
||||||
elseif needs_comma then
|
elseif needs_comma then
|
||||||
lua:add(" ")
|
lua:add(" ")
|
||||||
end
|
end
|
||||||
@ -417,6 +417,8 @@ local compile = setmetatable({
|
|||||||
if tree[i].type == "Comment" then
|
if tree[i].type == "Comment" then
|
||||||
items_lua:add("\n")
|
items_lua:add("\n")
|
||||||
sep = ''
|
sep = ''
|
||||||
|
elseif items_lua:trailing_line_len() > MAX_LINE then
|
||||||
|
sep = ',\n '
|
||||||
else
|
else
|
||||||
sep = ', '
|
sep = ', '
|
||||||
end
|
end
|
||||||
|
@ -234,7 +234,7 @@ compile = setmetatable({
|
|||||||
entry_lua\add as_lua(v)
|
entry_lua\add as_lua(v)
|
||||||
if needs_comma then lua\add ","
|
if needs_comma then lua\add ","
|
||||||
if lua\trailing_line_len! + #(entry_lua\text!\match("^[\n]*")) > MAX_LINE
|
if lua\trailing_line_len! + #(entry_lua\text!\match("^[\n]*")) > MAX_LINE
|
||||||
lua\add "\n"
|
lua\add "\n "
|
||||||
elseif needs_comma
|
elseif needs_comma
|
||||||
lua\add " "
|
lua\add " "
|
||||||
lua\add entry_lua
|
lua\add entry_lua
|
||||||
@ -322,6 +322,8 @@ compile = setmetatable({
|
|||||||
if tree[i].type == "Comment"
|
if tree[i].type == "Comment"
|
||||||
items_lua\add "\n"
|
items_lua\add "\n"
|
||||||
sep = ''
|
sep = ''
|
||||||
|
elseif items_lua\trailing_line_len! > MAX_LINE
|
||||||
|
sep = ',\n '
|
||||||
else
|
else
|
||||||
sep = ', '
|
sep = ', '
|
||||||
i += 1
|
i += 1
|
||||||
|
@ -26,6 +26,13 @@ do
|
|||||||
_with_0.unpack = unpack or table.unpack
|
_with_0.unpack = unpack or table.unpack
|
||||||
_with_0["nil"] = Cc(nil)
|
_with_0["nil"] = Cc(nil)
|
||||||
_with_0.userdata = Carg(1)
|
_with_0.userdata = Carg(1)
|
||||||
|
_with_0.indentation = lpeg.Cmt(P(0), function(s, i)
|
||||||
|
local sub = string.sub
|
||||||
|
while i > 1 and sub(s, i - 1, i - 1) ~= '\n' do
|
||||||
|
i = i - 1
|
||||||
|
end
|
||||||
|
return true, (s:match("^ *", i))
|
||||||
|
end)
|
||||||
_with_0.utf8_char = (R("\194\223") * R("\128\191") + R("\224\239") * R("\128\191") * R("\128\191") + R("\240\244") * R("\128\191") * R("\128\191") * R("\128\191"))
|
_with_0.utf8_char = (R("\194\223") * R("\128\191") + R("\224\239") * R("\128\191") * R("\128\191") + R("\240\244") * R("\128\191") * R("\128\191") * R("\128\191"))
|
||||||
_with_0.Tree = function(t, userdata)
|
_with_0.Tree = function(t, userdata)
|
||||||
return userdata.make_tree(t, userdata)
|
return userdata.make_tree(t, userdata)
|
||||||
|
@ -25,6 +25,13 @@ DEFS = with {}
|
|||||||
.unpack = unpack or table.unpack
|
.unpack = unpack or table.unpack
|
||||||
.nil = Cc(nil)
|
.nil = Cc(nil)
|
||||||
.userdata = Carg(1)
|
.userdata = Carg(1)
|
||||||
|
-- Always match and capture the indentation (spaces only) of the current line
|
||||||
|
-- i.e. the leading space of the chunk of non-newline characters leading up to s[i]
|
||||||
|
.indentation = lpeg.Cmt P(0),
|
||||||
|
(s, i)->
|
||||||
|
sub = string.sub
|
||||||
|
while i > 1 and sub(s,i-1,i-1) != '\n' do i -= 1
|
||||||
|
return true, (s\match("^ *", i))
|
||||||
.utf8_char = (
|
.utf8_char = (
|
||||||
R("\194\223")*R("\128\191") +
|
R("\194\223")*R("\128\191") +
|
||||||
R("\224\239")*R("\128\191")*R("\128\191") +
|
R("\224\239")*R("\128\191")*R("\128\191") +
|
||||||
|
@ -42,25 +42,44 @@ repeat:
|
|||||||
|
|
||||||
if ((size of $buff) == 0): stop
|
if ((size of $buff) == 0): stop
|
||||||
$buff = ($buff, joined)
|
$buff = ($buff, joined)
|
||||||
|
|
||||||
# TODO: support local variables
|
|
||||||
spoof file $buff
|
spoof file $buff
|
||||||
try:
|
try:
|
||||||
$ret = (run $buff)
|
$tree = ($buff parsed)
|
||||||
..and if it barfs $err: say $err
|
..and if it barfs $err:
|
||||||
..or if it succeeds:
|
say $err
|
||||||
if (type of $ret) is:
|
do next
|
||||||
"nil":
|
|
||||||
do nothing
|
|
||||||
|
|
||||||
"boolean":
|
unless $tree:
|
||||||
say "= \("yes" if $ret else "no")"
|
do next
|
||||||
|
|
||||||
"table":
|
for $chunk in $tree:
|
||||||
if $ret.as_nomsu:
|
try:
|
||||||
say "= \($ret, as nomsu)"
|
$lua = ($chunk as lua)
|
||||||
..else:
|
..and if it barfs $err: say $err
|
||||||
|
|
||||||
|
unless $lua:
|
||||||
|
do next
|
||||||
|
|
||||||
|
# TODO: this is a bit hacky, it just defaults variables to global
|
||||||
|
so that stuff mostly works across multiple lines. It would be
|
||||||
|
nicer if local variables actually worked.
|
||||||
|
$lua, remove free vars
|
||||||
|
try:
|
||||||
|
$ret = (run $lua)
|
||||||
|
..and if it barfs $err: say $err
|
||||||
|
..or if it succeeds:
|
||||||
|
if (type of $ret) is:
|
||||||
|
"nil":
|
||||||
|
do nothing
|
||||||
|
|
||||||
|
"boolean":
|
||||||
|
say "= \("yes" if $ret else "no")"
|
||||||
|
|
||||||
|
"table":
|
||||||
|
if $ret.as_nomsu:
|
||||||
|
say "= \($ret, as nomsu)"
|
||||||
|
..else:
|
||||||
|
say "= \$ret"
|
||||||
|
|
||||||
|
else:
|
||||||
say "= \$ret"
|
say "= \$ret"
|
||||||
|
|
||||||
else:
|
|
||||||
say "= \$ret"
|
|
||||||
|
Loading…
Reference in New Issue
Block a user