816 lines
22 KiB
Lua
816 lines
22 KiB
Lua
local utils = require('utils')
|
|
local repr, stringify, min, max, equivalent, set, is_list, sum
|
|
repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum
|
|
local immutable = require('immutable')
|
|
local insert, remove, concat
|
|
do
|
|
local _obj_0 = table
|
|
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
|
end
|
|
local Lua, Nomsu, Location
|
|
do
|
|
local _obj_0 = require("code_obj")
|
|
Lua, Nomsu, Location = _obj_0.Lua, _obj_0.Nomsu, _obj_0.Location
|
|
end
|
|
local MAX_LINE = 80
|
|
local Types = { }
|
|
Types.is_node = function(n)
|
|
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
|
end
|
|
local Tree
|
|
Tree = function(name, kind, methods)
|
|
assert((kind == 'single') or (kind == 'multi'))
|
|
local is_multi = (kind == 'multi')
|
|
do
|
|
methods.with_value = function(self, value)
|
|
return getmetatable(self)(value)
|
|
end
|
|
methods.type = name
|
|
methods.name = name
|
|
methods.is_multi = is_multi
|
|
if is_multi then
|
|
methods.__tostring = function(self)
|
|
return tostring(self.name) .. "(" .. tostring(table.concat((function()
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
for _index_0 = 1, #self do
|
|
local v = self[_index_0]
|
|
_accum_0[_len_0] = repr(v)
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
return _accum_0
|
|
end)(), ', ')) .. ")"
|
|
end
|
|
methods.map = function(self, fn)
|
|
do
|
|
local ret = fn(self)
|
|
if ret then
|
|
return ret
|
|
end
|
|
end
|
|
local new_vals
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
for _index_0 = 1, #self do
|
|
local v = self[_index_0]
|
|
_accum_0[_len_0] = v.map and v:map(fn) or v
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
new_vals = _accum_0
|
|
end
|
|
local ret = getmetatable(self)(unpack(new_vals))
|
|
return ret
|
|
end
|
|
else
|
|
methods.__tostring = function(self)
|
|
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
|
|
end
|
|
methods.map = function(self, fn)
|
|
return fn(self) or self
|
|
end
|
|
end
|
|
end
|
|
if is_multi then
|
|
Types[name] = immutable(nil, methods)
|
|
else
|
|
Types[name] = immutable({
|
|
"value"
|
|
}, methods)
|
|
end
|
|
end
|
|
Tree("EscapedNomsu", 'single', {
|
|
as_lua = function(self, nomsu)
|
|
local make_tree
|
|
make_tree = function(t)
|
|
if type(t) ~= 'userdata' then
|
|
return repr(t)
|
|
end
|
|
if t.is_multi then
|
|
local bits
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
for _index_0 = 1, #t do
|
|
local bit = t[_index_0]
|
|
_accum_0[_len_0] = make_tree(bit)
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
bits = _accum_0
|
|
end
|
|
return t.type .. "(" .. table.concat(bits, ", ") .. ")"
|
|
else
|
|
return t.type .. "(" .. make_tree(t.value) .. ")"
|
|
end
|
|
end
|
|
return Lua.Value(nil, make_tree(self.value))
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
local nomsu = self.value:as_nomsu(true)
|
|
if nomsu == nil and not inline then
|
|
nomsu = self.value:as_nomsu()
|
|
return nomsu and Nomsu(nil, "\\:\n ", nomsu)
|
|
end
|
|
return nomsu and Nomsu(nil, "\\(", nomsu, ")")
|
|
end,
|
|
map = function(self, fn)
|
|
return fn(self) or self:map(fn)
|
|
end
|
|
})
|
|
Tree("Block", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua()
|
|
for i, line in ipairs(self) do
|
|
local line_lua = line:as_lua(nomsu)
|
|
if i > 1 then
|
|
lua:append("\n")
|
|
end
|
|
lua:append(line_lua:as_statements())
|
|
end
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
local nomsu = Nomsu()
|
|
for i, line in ipairs(self) do
|
|
if i > 1 then
|
|
nomsu:append("; ")
|
|
end
|
|
local line_nomsu = line:as_nomsu(true)
|
|
if not (line_nomsu) then
|
|
return nil
|
|
end
|
|
nomsu:append(line_nomsu)
|
|
end
|
|
return nomsu
|
|
end
|
|
local nomsu = Nomsu()
|
|
for i, line in ipairs(self) do
|
|
line = assert(line:as_nomsu(nil, true), "Could not convert line to nomsu")
|
|
nomsu:append(line)
|
|
if i < #self then
|
|
nomsu:append("\n")
|
|
if tostring(line):match("\n") then
|
|
nomsu:append("\n")
|
|
end
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
})
|
|
local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]])
|
|
Tree("Action", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local stub = self:get_stub()
|
|
local compile_action = nomsu.environment.COMPILE_ACTIONS[stub]
|
|
if compile_action then
|
|
local args
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
for _index_0 = 1, #self do
|
|
local arg = self[_index_0]
|
|
if arg.type ~= "Word" then
|
|
_accum_0[_len_0] = arg
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
end
|
|
args = _accum_0
|
|
end
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
local _list_0 = nomsu.environment.ARG_ORDERS[compile_action][stub]
|
|
for _index_0 = 1, #_list_0 do
|
|
local p = _list_0[_index_0]
|
|
_accum_0[_len_0] = args[p - 1]
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
args = _accum_0
|
|
end
|
|
local ret = compile_action(self, unpack(args))
|
|
if not ret then
|
|
error("Failed to produce any Lua")
|
|
end
|
|
return ret
|
|
end
|
|
local action = rawget(nomsu.environment.ACTIONS, stub)
|
|
local lua = Lua.Value()
|
|
if not action and math_expression:match(stub) then
|
|
for i, tok in ipairs(self) do
|
|
if tok.type == "Word" then
|
|
lua:append(tok.value)
|
|
else
|
|
local tok_lua = tok:as_lua(nomsu)
|
|
if not (tok_lua.is_value) then
|
|
error("non-expression value inside math expression: " .. tostring(colored.yellow(repr(tok))))
|
|
end
|
|
if tok.type == "Action" then
|
|
tok_lua:parenthesize()
|
|
end
|
|
lua:append(tok_lua)
|
|
end
|
|
if i < #self then
|
|
lua:append(" ")
|
|
end
|
|
end
|
|
return lua
|
|
end
|
|
local args = { }
|
|
for i, tok in ipairs(self) do
|
|
local _continue_0 = false
|
|
repeat
|
|
if tok.type == "Word" then
|
|
_continue_0 = true
|
|
break
|
|
end
|
|
local arg_lua = tok:as_lua(nomsu)
|
|
if not (arg_lua.is_value) then
|
|
error("Cannot use:\n" .. tostring(colored.yellow(repr(tok))) .. "\nas an argument to " .. tostring(stub) .. ", since it's not an expression, it produces: " .. tostring(repr(arg_lua)), 0)
|
|
end
|
|
insert(args, arg_lua)
|
|
_continue_0 = true
|
|
until true
|
|
if not _continue_0 then
|
|
break
|
|
end
|
|
end
|
|
if action then
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
local _list_0 = nomsu.environment.ARG_ORDERS[action][stub]
|
|
for _index_0 = 1, #_list_0 do
|
|
local p = _list_0[_index_0]
|
|
_accum_0[_len_0] = args[p]
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
args = _accum_0
|
|
end
|
|
end
|
|
lua:append("ACTIONS[", repr(stub), "](")
|
|
for i, arg in ipairs(args) do
|
|
lua:append(arg)
|
|
if i < #args then
|
|
lua:append(", ")
|
|
end
|
|
end
|
|
lua:append(")")
|
|
return lua
|
|
end,
|
|
get_stub = function(self, include_names)
|
|
if include_names == nil then
|
|
include_names = false
|
|
end
|
|
local bits
|
|
if include_names then
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
for _index_0 = 1, #self do
|
|
local t = self[_index_0]
|
|
_accum_0[_len_0] = (t.type == "Word" and t.value or "%" .. tostring(t.value))
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
bits = _accum_0
|
|
end
|
|
else
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
for _index_0 = 1, #self do
|
|
local t = self[_index_0]
|
|
_accum_0[_len_0] = (t.type == "Word" and t.value or "%")
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
bits = _accum_0
|
|
end
|
|
end
|
|
return concat(bits, " ")
|
|
end,
|
|
as_nomsu = function(self, inline, can_use_colon)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if can_use_colon == nil then
|
|
can_use_colon = false
|
|
end
|
|
if inline then
|
|
local nomsu = Nomsu()
|
|
for i, bit in ipairs(self) do
|
|
if bit.type == "Word" then
|
|
if i > 1 then
|
|
nomsu:append(" ")
|
|
end
|
|
nomsu:append(bit.value)
|
|
else
|
|
local arg_nomsu = bit:as_nomsu(true)
|
|
if not (arg_nomsu) then
|
|
return nil
|
|
end
|
|
if not (i == 1) then
|
|
nomsu:append(" ")
|
|
end
|
|
if bit.type == "Action" or bit.type == "Block" then
|
|
arg_nomsu:parenthesize()
|
|
end
|
|
nomsu:append(arg_nomsu)
|
|
end
|
|
end
|
|
return nomsu
|
|
else
|
|
local nomsu = Nomsu()
|
|
local next_space = ""
|
|
local last_colon = nil
|
|
for i, bit in ipairs(self) do
|
|
if bit.type == "Word" then
|
|
nomsu:append(next_space, bit.value)
|
|
next_space = " "
|
|
else
|
|
local arg_nomsu
|
|
if last_colon == i - 1 and bit.type == "Action" then
|
|
arg_nomsu = nil
|
|
elseif bit.type == "Block" then
|
|
arg_nomsu = nil
|
|
else
|
|
arg_nomsu = bit:as_nomsu(true)
|
|
end
|
|
if arg_nomsu and #arg_nomsu < MAX_LINE then
|
|
if bit.type == "Action" then
|
|
if can_use_colon and i > 1 then
|
|
nomsu:append(next_space:match("[^ ]*"), ": ", arg_nomsu)
|
|
next_space = "\n.."
|
|
last_colon = i
|
|
else
|
|
nomsu:append(next_space, "(", arg_nomsu, ")")
|
|
next_space = " "
|
|
end
|
|
else
|
|
nomsu:append(next_space, arg_nomsu)
|
|
next_space = " "
|
|
end
|
|
else
|
|
arg_nomsu = bit:as_nomsu(nil, true)
|
|
if not (nomsu) then
|
|
return nil
|
|
end
|
|
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
|
if i == 1 then
|
|
arg_nomsu = Nomsu(nil, "(..)\n ", arg_nomsu)
|
|
else
|
|
arg_nomsu = Nomsu(nil, "\n ", arg_nomsu)
|
|
end
|
|
end
|
|
if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then
|
|
next_space = ""
|
|
end
|
|
nomsu:append(next_space, arg_nomsu)
|
|
next_space = "\n.."
|
|
end
|
|
if next_space == " " and #(tostring(nomsu):match("[^\n]*$")) > MAX_LINE then
|
|
next_space = "\n.."
|
|
end
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("Text", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua.Value()
|
|
local string_buffer = ""
|
|
for _index_0 = 1, #self do
|
|
local _continue_0 = false
|
|
repeat
|
|
local bit = self[_index_0]
|
|
if type(bit) == "string" then
|
|
string_buffer = string_buffer .. bit
|
|
_continue_0 = true
|
|
break
|
|
end
|
|
if string_buffer ~= "" then
|
|
if #lua.bits > 0 then
|
|
lua:append("..")
|
|
end
|
|
lua:append(repr(string_buffer))
|
|
string_buffer = ""
|
|
end
|
|
local bit_lua = bit:as_lua(nomsu)
|
|
if not (bit_lua.is_value) then
|
|
error("Cannot use " .. tostring(colored.yellow(repr(bit))) .. " as a string interpolation value, since it's not an expression.", 0)
|
|
end
|
|
if #lua.bits > 0 then
|
|
lua:append("..")
|
|
end
|
|
if bit.type ~= "Text" then
|
|
bit_lua = Lua.Value(nil, "stringify(", bit_lua, ")")
|
|
end
|
|
lua:append(bit_lua)
|
|
_continue_0 = true
|
|
until true
|
|
if not _continue_0 then
|
|
break
|
|
end
|
|
end
|
|
if string_buffer ~= "" or #lua.bits == 0 then
|
|
if #lua.bits > 0 then
|
|
lua:append("..")
|
|
end
|
|
lua:append(repr(string_buffer))
|
|
end
|
|
if #lua.bits > 1 then
|
|
lua:parenthesize()
|
|
end
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
local nomsu = Nomsu(nil, '"')
|
|
for _index_0 = 1, #self do
|
|
local bit = self[_index_0]
|
|
if type(bit) == 'string' then
|
|
nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\\n")))
|
|
else
|
|
local interp_nomsu = bit:as_nomsu(true)
|
|
if interp_nomsu then
|
|
if bit.type ~= "Word" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
|
interp_nomsu:parenthesize()
|
|
end
|
|
nomsu:append("\\", interp_nomsu)
|
|
else
|
|
return nil
|
|
end
|
|
end
|
|
end
|
|
nomsu:append('"')
|
|
return nomsu
|
|
else
|
|
local inline_version = self:as_nomsu(true)
|
|
if inline_version and #inline_version <= MAX_LINE then
|
|
return inline_version
|
|
end
|
|
local nomsu = Nomsu(nil, '".."\n ')
|
|
for i, bit in ipairs(self) do
|
|
if type(bit) == 'string' then
|
|
nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\n ")))
|
|
else
|
|
local interp_nomsu = bit:as_nomsu(true)
|
|
if interp_nomsu then
|
|
if bit.type ~= "Word" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
|
interp_nomsu:parenthesize()
|
|
end
|
|
nomsu:append("\\", interp_nomsu)
|
|
else
|
|
interp_nomsu = bit:as_nomsu()
|
|
if not (interp_nomsu) then
|
|
return nil
|
|
end
|
|
nomsu:append("\\\n ", interp_nomsu)
|
|
if i < #self then
|
|
nomsu:append("\n ..")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("List", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua.Value(nil, "{")
|
|
local line_length = 0
|
|
for i, item in ipairs(self) do
|
|
local item_lua = item:as_lua(nomsu)
|
|
if not (item_lua.is_value) then
|
|
error("Cannot use " .. tostring(colored.yellow(repr(item))) .. " as a list item, since it's not an expression.", 0)
|
|
end
|
|
lua:append(item_lua)
|
|
local item_string = tostring(item_lua)
|
|
local last_line = item_string:match("[^\n]*$")
|
|
if item_string:match("\n") then
|
|
line_length = #last_line
|
|
else
|
|
line_length = line_length + #last_line
|
|
end
|
|
if i < #self then
|
|
if line_length >= MAX_LINE then
|
|
lua:append(",\n ")
|
|
line_length = 0
|
|
else
|
|
lua:append(", ")
|
|
line_length = line_length + 2
|
|
end
|
|
end
|
|
end
|
|
lua:append("}")
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
local nomsu = Nomsu(nil, "[")
|
|
for i, item in ipairs(self) do
|
|
local item_nomsu = item:as_nomsu(true)
|
|
if not (item_nomsu) then
|
|
return nil
|
|
end
|
|
if i > 1 then
|
|
nomsu:append(", ")
|
|
end
|
|
nomsu:append(item_nomsu)
|
|
end
|
|
nomsu:append("]")
|
|
return nomsu
|
|
else
|
|
local inline_version = self:as_nomsu(true)
|
|
if inline_version and #inline_version <= MAX_LINE then
|
|
return inline_version
|
|
end
|
|
local nomsu = Nomsu(nil, "[..]")
|
|
local line = Nomsu(nil, "\n ")
|
|
for _index_0 = 1, #self do
|
|
local item = self[_index_0]
|
|
local item_nomsu = item:as_nomsu(true)
|
|
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then
|
|
if #line.bits > 1 then
|
|
line:append(", ")
|
|
end
|
|
line:append(item_nomsu)
|
|
else
|
|
if not (item_nomsu) then
|
|
item_nomsu = item:as_nomsu()
|
|
if not (item_nomsu) then
|
|
return nil
|
|
end
|
|
end
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
line = Nomsu(nil, "\n ")
|
|
end
|
|
line:append(item_nomsu)
|
|
end
|
|
end
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("Dict", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua.Value(nil, "{")
|
|
local line_length = 0
|
|
for i, entry in ipairs(self) do
|
|
local entry_lua = entry:as_lua(nomsu)
|
|
lua:append(entry_lua)
|
|
local entry_lua_str = tostring(entry_lua)
|
|
local last_line = entry_lua_str:match("\n([^\n]*)$")
|
|
if last_line then
|
|
line_length = #last_line
|
|
else
|
|
line_length = line_length + #entry_lua_str
|
|
end
|
|
if i < #self then
|
|
if line_length >= MAX_LINE then
|
|
lua:append(",\n ")
|
|
line_length = 0
|
|
else
|
|
lua:append(", ")
|
|
line_length = line_length + 2
|
|
end
|
|
end
|
|
end
|
|
lua:append("}")
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
local nomsu = Nomsu(nil, "{")
|
|
for i, entry in ipairs(self) do
|
|
local entry_nomsu = entry:as_nomsu(true)
|
|
if not (entry_nomsu) then
|
|
return nil
|
|
end
|
|
if i > 1 then
|
|
nomsu:append(", ")
|
|
end
|
|
nomsu:append(entry_nomsu)
|
|
end
|
|
nomsu:append("}")
|
|
return nomsu
|
|
else
|
|
local inline_version = self:as_nomsu(true)
|
|
if inline_version then
|
|
return inline_version
|
|
end
|
|
local nomsu = Nomsu(nil, "{..}")
|
|
local line = Nomsu(nil, "\n ")
|
|
for _index_0 = 1, #self do
|
|
local entry = self[_index_0]
|
|
local entry_nomsu = entry:as_nomsu()
|
|
if not (entry_nomsu) then
|
|
return nil
|
|
end
|
|
if #line + #tostring(entry_nomsu) <= MAX_LINE then
|
|
if #line.bits > 1 then
|
|
line:append(", ")
|
|
end
|
|
line:append(entry_nomsu)
|
|
else
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
line = Nomsu(nil, "\n ")
|
|
end
|
|
line:append(entry_nomsu)
|
|
end
|
|
end
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("DictEntry", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local key, value = self[1], self[2]
|
|
local key_lua = key:as_lua(nomsu)
|
|
if not (key_lua.is_value) then
|
|
error("Cannot use " .. tostring(colored.yellow(repr(key))) .. " as a dict key, since it's not an expression.", 0)
|
|
end
|
|
local value_lua = value and value:as_lua(nomsu) or Lua.Value(nil, "true")
|
|
if not (value_lua.is_value) then
|
|
error("Cannot use " .. tostring(colored.yellow(repr(value))) .. " as a dict value, since it's not an expression.", 0)
|
|
end
|
|
local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
|
if key_str then
|
|
return Lua(nil, key_str, "=", value_lua)
|
|
elseif tostring(key_lua):sub(1, 1) == "[" then
|
|
return Lua(nil, "[ ", key_lua, "]=", value_lua)
|
|
else
|
|
return Lua(nil, "[", key_lua, "]=", value_lua)
|
|
end
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = true
|
|
end
|
|
local key, value = self[1], self[2]
|
|
local key_nomsu = key:as_nomsu(true)
|
|
if not (key_nomsu) then
|
|
return nil
|
|
end
|
|
if key.type == "Action" or key.type == "Block" then
|
|
key_nomsu:parenthesize()
|
|
end
|
|
local value_nomsu
|
|
if value then
|
|
value_nomsu = value:as_nomsu(true)
|
|
else
|
|
value_nomsu = Nomsu(nil, "")
|
|
end
|
|
if inline and not value_nomsu then
|
|
return nil
|
|
end
|
|
if not value_nomsu then
|
|
if inline then
|
|
return nil
|
|
end
|
|
value_nomsu = value:as_nomsu()
|
|
if not (value_nomsu) then
|
|
return nil
|
|
end
|
|
end
|
|
return Nomsu(nil, key_nomsu, ":", value_nomsu)
|
|
end
|
|
})
|
|
Tree("IndexChain", 'multi', {
|
|
as_lua = function(self, nomsu)
|
|
local lua = self[1]:as_lua(nomsu)
|
|
if not (lua.is_value) then
|
|
error("Cannot index " .. tostring(colored.yellow(repr(self[1]))) .. ", since it's not an expression.", 0)
|
|
end
|
|
local first_char = tostring(lua):sub(1, 1)
|
|
if first_char == "{" or first_char == '"' or first_char == "[" then
|
|
lua:parenthesize()
|
|
end
|
|
for i = 2, #self do
|
|
local key = self[i]
|
|
local key_lua = key:as_lua(nomsu)
|
|
if not (key_lua.is_value) then
|
|
error("Cannot use " .. tostring(colored.yellow(repr(key))) .. " as an index, since it's not an expression.", 0)
|
|
end
|
|
local key_lua_str = tostring(key_lua)
|
|
do
|
|
local lua_id = key_lua_str:match("^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
|
if lua_id then
|
|
lua:append("." .. tostring(lua_id))
|
|
elseif key_lua_str:sub(1, 1) == '[' then
|
|
lua:append("[ ", key_lua, " ]")
|
|
else
|
|
lua:append("[", key_lua, "]")
|
|
end
|
|
end
|
|
end
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
local nomsu = Nomsu()
|
|
for i, bit in ipairs(self) do
|
|
if i > 1 then
|
|
nomsu:append(".")
|
|
end
|
|
local bit_nomsu = bit:as_nomsu(true)
|
|
if not (bit_nomsu) then
|
|
return nil
|
|
end
|
|
if bit.type == "Action" or bit.type == "Block" then
|
|
bit_nomsu:parenthesize()
|
|
end
|
|
nomsu:append(bit_nomsu)
|
|
end
|
|
return nomsu
|
|
end
|
|
})
|
|
Tree("Number", 'single', {
|
|
as_lua = function(self, nomsu)
|
|
return Lua.Value(nil, tostring(self.value))
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
return Nomsu(nil, tostring(self.value))
|
|
end
|
|
})
|
|
Tree("Var", 'single', {
|
|
as_lua_id = function(v)
|
|
return "_" .. (v:gsub("%W", function(c)
|
|
if c == "_" then
|
|
return "__"
|
|
else
|
|
return ("_%x"):format(c:byte())
|
|
end
|
|
end))
|
|
end,
|
|
as_lua = function(self, nomsu)
|
|
return Lua.Value(nil, self.as_lua_id(self.value))
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
return Nomsu(nil, "%", self.value)
|
|
end
|
|
})
|
|
Tree("Word", 'single', {
|
|
as_lua = function(self, nomsu)
|
|
return error("Attempt to convert Word to lua")
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
return Nomsu(nil, self.value)
|
|
end
|
|
})
|
|
Tree("Comment", 'single', {
|
|
as_lua = function(self, nomsu)
|
|
return Lua(nil, "--" .. self.value:gsub("\n", "\n--") .. "\n")
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
return nil
|
|
end
|
|
if self.value:match("\n") then
|
|
return Nomsu(nil, "#..", self.value:gsub("\n", "\n "))
|
|
else
|
|
return Nomsu(nil, "#", self.value)
|
|
end
|
|
end
|
|
})
|
|
return Types
|