787 lines
23 KiB
Lua
787 lines
23 KiB
Lua
local utils = require('utils')
|
|
local re = require('re')
|
|
local lpeg = require('lpeg')
|
|
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.DictEntry = immutable({
|
|
"key",
|
|
"value"
|
|
}, {
|
|
name = "DictEntry"
|
|
})
|
|
Types.is_node = function(n)
|
|
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
|
end
|
|
local Tree
|
|
Tree = function(name, methods)
|
|
do
|
|
methods.__tostring = function(self)
|
|
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ", " .. tostring(repr(self.source)) .. ")"
|
|
end
|
|
methods.with_value = function(self, value)
|
|
return getmetatable(self)(value, self.source)
|
|
end
|
|
methods.type = name
|
|
methods.name = name
|
|
methods.original_nomsu = function(self)
|
|
local leading_space = 0
|
|
local src_file = FILE_CACHE[self.source.filename]
|
|
while src_file:sub(self.source.start - leading_space - 1, self.source.start - leading_space - 1) == " " do
|
|
leading_space = leading_space + 1
|
|
end
|
|
if src_file:sub(self.source.start - leading_space - 1, self.source.start - leading_space - 1) ~= "\n" then
|
|
leading_space = 0
|
|
end
|
|
local ret = tostring(self.source:get_text()):gsub("\n" .. ((" "):rep(leading_space)), "\n")
|
|
return ret
|
|
end
|
|
end
|
|
Types[name] = immutable({
|
|
"value",
|
|
"source"
|
|
}, methods)
|
|
end
|
|
Tree("File", {
|
|
as_lua = function(self, nomsu)
|
|
if #self.value == 1 then
|
|
return self.value[1]:as_lua(nomsu)
|
|
end
|
|
local lua = Lua(self.source)
|
|
for i, line in ipairs(self.value) do
|
|
local line_lua = line:as_lua(nomsu)
|
|
if not line_lua then
|
|
error("No lua produced by " .. tostring(repr(line)), 0)
|
|
end
|
|
if i > 1 then
|
|
lua:append("\n")
|
|
end
|
|
lua:convert_to_statements()
|
|
lua:append(line_lua)
|
|
end
|
|
lua:declare_locals()
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
return nil
|
|
end
|
|
local nomsu = Nomsu(self.source)
|
|
for i, line in ipairs(self.value) do
|
|
line = assert(line:as_nomsu(nil, true), "Could not convert line to nomsu")
|
|
nomsu:append(line)
|
|
if i < #self.value then
|
|
if tostring(line):match("\n") then
|
|
nomsu:append("\n")
|
|
end
|
|
nomsu:append("\n")
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
})
|
|
Tree("Nomsu", {
|
|
as_lua = function(self, nomsu)
|
|
return Lua.Value(self.source, "nomsu:parse(Nomsu(", repr(self.value.source), ", ", repr(tostring(self.value.source:get_text())), ")).value[1]")
|
|
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(self.source, "\\:\n ", nomsu)
|
|
end
|
|
return nomsu and Nomsu(self.source, "\\(", nomsu, ")")
|
|
end
|
|
})
|
|
Tree("Block", {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua(self.source)
|
|
for i, line in ipairs(self.value) do
|
|
local line_lua = line:as_lua(nomsu)
|
|
if i > 1 then
|
|
lua:append("\n")
|
|
end
|
|
line_lua:convert_to_statements()
|
|
lua:append(line_lua)
|
|
end
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
if inline then
|
|
local nomsu = Nomsu(self.source)
|
|
for i, line in ipairs(self.value) 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(self.source)
|
|
for i, line in ipairs(self.value) do
|
|
line = assert(line:as_nomsu(nil, true), "Could not convert line to nomsu")
|
|
nomsu:append(line)
|
|
if i < #self.value then
|
|
nomsu:append("\n")
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
})
|
|
local math_expression = re.compile([[ "%" (" " [*/^+-] " %")+ ]])
|
|
Tree("Action", {
|
|
as_lua = function(self, nomsu)
|
|
local stub = self:get_stub()
|
|
local action = rawget(nomsu.environment.ACTIONS, stub)
|
|
local metadata = nomsu.action_metadata[action]
|
|
if metadata and metadata.compile_time then
|
|
local args
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local arg = _list_0[_index_0]
|
|
if arg.type ~= "Word" then
|
|
_accum_0[_len_0] = arg
|
|
_len_0 = _len_0 + 1
|
|
end
|
|
end
|
|
args = _accum_0
|
|
end
|
|
if metadata.arg_orders then
|
|
local new_args
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
local _list_0 = metadata.arg_orders[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
|
|
new_args = _accum_0
|
|
end
|
|
args = new_args
|
|
end
|
|
local ret = action(self, unpack(args))
|
|
return ret
|
|
end
|
|
local lua = Lua.Value(self.source)
|
|
if not metadata and math_expression:match(stub) then
|
|
for i, tok in ipairs(self.value) 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
|
|
local src = tok.source:get_text()
|
|
error("non-expression value inside math expression: " .. tostring(colored.yellow(src)))
|
|
end
|
|
lua:append(tok_lua)
|
|
end
|
|
if i < #self.value then
|
|
lua:append(" ")
|
|
end
|
|
end
|
|
lua:parenthesize()
|
|
return lua
|
|
end
|
|
local args = { }
|
|
for i, tok in ipairs(self.value) 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
|
|
local line, src = tok.source:get_line(), tok.source:get_text()
|
|
error(tostring(line) .. ": Cannot use:\n" .. tostring(colored.yellow(src)) .. "\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 metadata and metadata.arg_orders then
|
|
do
|
|
local _accum_0 = { }
|
|
local _len_0 = 1
|
|
local _list_0 = metadata.arg_orders[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
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local t = _list_0[_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
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local t = _list_0[_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(self.source)
|
|
for i, bit in ipairs(self.value) 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(self.source)
|
|
local next_space = ""
|
|
local last_colon = nil
|
|
for i, bit in ipairs(self.value) do
|
|
if bit.type == "Word" then
|
|
nomsu:append(next_space, bit.value)
|
|
next_space = " "
|
|
else
|
|
local arg_nomsu = bit.type ~= "Block" and bit:as_nomsu(true)
|
|
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(bit.source, "(..)\n ", arg_nomsu)
|
|
else
|
|
arg_nomsu = Nomsu(bit.source, "\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
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("Text", {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua.Value(self.source)
|
|
local string_buffer = ""
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local _continue_0 = false
|
|
repeat
|
|
local bit = _list_0[_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
|
|
local line, src = bit.source:get_line(), bit.source:get_text()
|
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(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(bit.source, "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(self.source, '"')
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local bit = _list_0[_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(self.source, '".."\n ')
|
|
for i, bit in ipairs(self.value) 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.value then
|
|
nomsu:append("\n ..")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("List", {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua.Value(self.source, "{")
|
|
local line_length = 0
|
|
for i, item in ipairs(self.value) do
|
|
local item_lua = item:as_lua(nomsu)
|
|
if not (item_lua.is_value) then
|
|
local line, src = item.source:get_line(), item.source:get_text()
|
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " 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.value 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(self.source, "[")
|
|
for i, item in ipairs(self.value) 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(self.source, "[..]")
|
|
local line = Nomsu(self.source, "\n ")
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local item = _list_0[_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(item.source, "\n ")
|
|
end
|
|
line:append(item_nomsu)
|
|
end
|
|
end
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("Dict", {
|
|
as_lua = function(self, nomsu)
|
|
local lua = Lua.Value(self.source, "{")
|
|
local line_length = 0
|
|
for i, entry in ipairs(self.value) do
|
|
local key_lua = entry.key:as_lua(nomsu)
|
|
if not (key_lua.is_value) then
|
|
local line, src = key.source:get_line(), key.source:get_text()
|
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict key, since it's not an expression.", 0)
|
|
end
|
|
local value_lua = entry.value:as_lua(nomsu)
|
|
if not (value_lua.is_value) then
|
|
local line, src = value.source:get_line(), value.source:get_text()
|
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " 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
|
|
lua:append(key_str, "=", value_lua)
|
|
elseif tostring(key_lua):sub(1, 1) == "[" then
|
|
lua:append("[ ", key_lua, "]=", value_lua)
|
|
else
|
|
lua:append("[", key_lua, "]=", value_lua)
|
|
end
|
|
local newlines, last_line = ("[" .. tostring(key_lua) .. "=" .. tostring(value_lua)):match("^(.-)([^\n]*)$")
|
|
if #newlines > 0 then
|
|
line_length = #last_line
|
|
else
|
|
line_length = line_length + #last_line
|
|
end
|
|
if i < #self.value 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(self.source, "{")
|
|
for i, entry in ipairs(self.value) do
|
|
local key_nomsu = entry.key:as_nomsu(true)
|
|
if not (key_nomsu) then
|
|
return nil
|
|
end
|
|
if entry.key.type == "Action" or entry.key.type == "Block" then
|
|
key_nomsu:parenthesize()
|
|
end
|
|
local value_nomsu = entry.value:as_nomsu(true)
|
|
if not (value_nomsu) then
|
|
return nil
|
|
end
|
|
if i > 1 then
|
|
nomsu:append(", ")
|
|
end
|
|
nomsu:append(key_nomsu, ":", value_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(self.source, "{..}")
|
|
local line = Nomsu(self.source, "\n ")
|
|
local _list_0 = self.value
|
|
for _index_0 = 1, #_list_0 do
|
|
local entry = _list_0[_index_0]
|
|
local key_nomsu = entry.key:as_nomsu(true)
|
|
if not (key_nomsu) then
|
|
return nil
|
|
end
|
|
if entry.key.type == "Action" or entry.key.type == "Block" then
|
|
key_nomsu:parenthesize()
|
|
end
|
|
local value_nomsu = entry.value:as_nomsu(true)
|
|
if value_nomsu and #line + #", " + #key_nomsu + #":" + #value_nomsu <= MAX_LINE then
|
|
if #line.bits > 1 then
|
|
line:append(", ")
|
|
end
|
|
line:append(key_nomsu, ":", value_nomsu)
|
|
else
|
|
if not (value_nomsu) then
|
|
value_nomsu = entry.value:as_nomsu()
|
|
if not (value_nomsu) then
|
|
return nil
|
|
end
|
|
end
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
line = Nomsu(bit.source, "\n ")
|
|
end
|
|
line:append(key_nomsu, ":", value_nomsu)
|
|
end
|
|
end
|
|
if #line.bits > 1 then
|
|
nomsu:append(line)
|
|
end
|
|
return nomsu
|
|
end
|
|
end
|
|
})
|
|
Tree("IndexChain", {
|
|
as_lua = function(self, nomsu)
|
|
local lua = self.value[1]:as_lua(nomsu)
|
|
if not (lua.is_value) then
|
|
local line, src = self.value[1].source:get_line(), self.value[1].source:get_text()
|
|
error(tostring(line) .. ": Cannot index " .. tostring(colored.yellow(src)) .. ", 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.value do
|
|
local _continue_0 = false
|
|
repeat
|
|
local key = self.value[i]
|
|
if key.type == 'Text' and #key.value == 1 and type(key.value[1]) == 'string' and key.value[1]:match("^[a-zA-Z_][a-zA-Z0-9_]$") then
|
|
lua:append("." .. tostring(key.value[1]))
|
|
_continue_0 = true
|
|
break
|
|
end
|
|
local key_lua = key:as_lua(nomsu)
|
|
if not (key_lua.is_value) then
|
|
local line, src = key.source:get_line(), key.source:get_text()
|
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as an index, since it's not an expression.", 0)
|
|
end
|
|
if tostring(key_lua):sub(1, 1) == '[' then
|
|
lua:append("[ ", key_lua, "]")
|
|
else
|
|
lua:append("[", key_lua, "]")
|
|
end
|
|
_continue_0 = true
|
|
until true
|
|
if not _continue_0 then
|
|
break
|
|
end
|
|
end
|
|
return lua
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
local nomsu = Nomsu(self.source)
|
|
for i, bit in ipairs(self.value) do
|
|
if i > 1 then
|
|
nomsu:append(".")
|
|
end
|
|
local bit_nomsu = bit:as_nomsu(true)
|
|
if not (bit_nomsu) then
|
|
return nil
|
|
end
|
|
nomsu:append(bit_nomsu)
|
|
end
|
|
return nomsu
|
|
end
|
|
})
|
|
Tree("Number", {
|
|
as_lua = function(self, nomsu)
|
|
return Lua.Value(self.source, tostring(self.value))
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
return Nomsu(self.source, tostring(self.value))
|
|
end
|
|
})
|
|
Tree("Var", {
|
|
as_lua = function(self, nomsu)
|
|
local lua_id = "_" .. (self.value:gsub("%W", function(verboten)
|
|
if verboten == "_" then
|
|
return "__"
|
|
else
|
|
return ("_%x"):format(verboten:byte())
|
|
end
|
|
end))
|
|
return Lua.Value(self.source, lua_id)
|
|
end,
|
|
as_nomsu = function(self, inline)
|
|
if inline == nil then
|
|
inline = false
|
|
end
|
|
return Nomsu(self.source, "%", self.value)
|
|
end
|
|
})
|
|
Tree("Word", {
|
|
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(self.source, self.value)
|
|
end
|
|
})
|
|
Tree("Comment", {
|
|
as_lua = function(self, nomsu)
|
|
return Lua(self.source, "--" .. 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(self.source, "#..", self.value:gsub("\n", "\n "))
|
|
else
|
|
return Nomsu(self.source, "#", self.value)
|
|
end
|
|
end
|
|
})
|
|
return Types
|