2018-06-19 01:27:32 -07:00
|
|
|
-- This file contains the source code of the Nomsu compiler.
|
|
|
|
-- Nomsu is a programming language that cross-compiles to Lua. It was designed to be good
|
|
|
|
-- at natural-language-like code that is highly self-modifying and flexible.
|
|
|
|
-- The only dependency is LPEG, which can be installed using "luarocks install lpeg"
|
|
|
|
-- File usage:
|
|
|
|
-- Either, in a lua/moonscript file:
|
|
|
|
-- Nomsu = require "nomsu"
|
|
|
|
-- nomsu = Nomsu()
|
|
|
|
-- nomsu:run(your_nomsu_code)
|
|
|
|
-- Or from the command line:
|
|
|
|
-- lua nomsu.lua your_file.nom
|
|
|
|
lpeg = require 'lpeg'
|
2018-09-14 14:38:59 -07:00
|
|
|
{:R,:P,:S} = lpeg
|
2018-06-19 01:27:32 -07:00
|
|
|
re = require 're'
|
2018-09-10 15:55:34 -07:00
|
|
|
{:List, :Dict, :Text} = require 'containers'
|
2018-06-19 01:27:32 -07:00
|
|
|
{:insert, :remove, :concat} = table
|
|
|
|
unpack or= table.unpack
|
2018-06-24 23:18:32 -07:00
|
|
|
{:match, :sub, :gsub, :format, :byte, :find} = string
|
2018-11-08 15:23:22 -08:00
|
|
|
{:LuaCode, :Source} = require "code_obj"
|
2018-10-31 15:54:18 -07:00
|
|
|
SyntaxTree = require "syntax_tree"
|
2018-11-08 15:23:22 -08:00
|
|
|
{:Importer, :import_to_1_from, :_1_forked} = require 'importer'
|
2018-11-17 14:38:05 -08:00
|
|
|
Files = require "files"
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-08-31 15:21:47 -07:00
|
|
|
table.map = (t, fn)-> setmetatable([fn(v) for _,v in ipairs(t)], getmetatable(t))
|
2018-09-14 14:38:59 -07:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
-- TODO: de-duplicate this
|
|
|
|
pretty_error = require("pretty_errors")
|
2018-11-17 14:38:05 -08:00
|
|
|
compile_error = (source, err_msg, hint=nil)->
|
|
|
|
local file
|
|
|
|
if SyntaxTree\is_instance(source)
|
|
|
|
file = source\get_source_file!
|
|
|
|
source = source.source
|
|
|
|
elseif type(source) == 'string'
|
|
|
|
source = Source\from_string(source)
|
|
|
|
if source and not file
|
|
|
|
file = Files.read(source.filename)
|
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
err_str = pretty_error{
|
|
|
|
title: "Compile error"
|
2018-11-17 14:38:05 -08:00
|
|
|
error:err_msg, hint:hint, source:file
|
|
|
|
start:source.start, stop:source.stop, filename:source.filename
|
2018-09-16 16:57:14 -07:00
|
|
|
}
|
2018-11-08 15:23:22 -08:00
|
|
|
error(err_str, 0)
|
|
|
|
|
|
|
|
-- This is a bit of a hack, but this code handles arbitrarily complex
|
|
|
|
-- math expressions like 2*x + 3^2 without having to define a single
|
|
|
|
-- action for every possibility.
|
|
|
|
math_expression = re.compile [[ (([*/^+-] / [0-9]+) " ")* [*/^+-] !. ]]
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
|
|
|
|
compile = setmetatable({
|
|
|
|
action: Importer{
|
2018-11-09 16:40:36 -08:00
|
|
|
[""]: (compile, fn, ...)->
|
|
|
|
lua = LuaCode!
|
2018-11-08 15:23:22 -08:00
|
|
|
fn_lua = compile(fn)
|
|
|
|
lua\append fn_lua
|
|
|
|
unless fn_lua\text!\match("^%(.*%)$") or fn_lua\text!\match("^[_a-zA-Z][_a-zA-Z0-9.]*$")
|
|
|
|
lua\parenthesize!
|
2018-11-06 15:13:55 -08:00
|
|
|
lua\append "("
|
|
|
|
for i=1,select('#',...)
|
|
|
|
lua\append(", ") if i > 1
|
2018-11-17 14:38:05 -08:00
|
|
|
lua\append compile((select(i, ...)))
|
2018-11-06 15:13:55 -08:00
|
|
|
lua\append ")"
|
|
|
|
return lua
|
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
["Lua"]: (compile, code)->
|
2018-11-19 17:21:08 -08:00
|
|
|
if not code
|
|
|
|
return LuaCode("LuaCode()")
|
2018-09-06 12:46:39 -07:00
|
|
|
if code.type != "Text"
|
2018-11-09 16:40:36 -08:00
|
|
|
return LuaCode("LuaCode:from(", tostring(code.source)\as_lua!, ", ", compile(code), ")")
|
2018-11-08 15:23:22 -08:00
|
|
|
add_bit_lua = (lua, bit_lua)->
|
|
|
|
bit_leading_len = #(bit_lua\match("^[^\n]*"))
|
|
|
|
lua\append(lua\trailing_line_len! + bit_leading_len > MAX_LINE and ",\n " or ", ")
|
|
|
|
lua\append(bit_lua)
|
|
|
|
operate_on_text = (text)->
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from(text.source, "LuaCode:from(", tostring(text.source)\as_lua!)
|
2018-11-08 15:23:22 -08:00
|
|
|
for bit in *text
|
|
|
|
if type(bit) == "string"
|
|
|
|
add_bit_lua(lua, bit\as_lua!)
|
|
|
|
elseif bit.type == "Text"
|
|
|
|
add_bit_lua(lua, operate_on_text(bit))
|
|
|
|
else
|
|
|
|
add_bit_lua(lua, compile(bit))
|
|
|
|
lua\append ")"
|
|
|
|
return lua
|
|
|
|
return operate_on_text code
|
2018-10-31 03:51:37 -07:00
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
["lua >"]: (compile, code)->
|
2018-11-08 15:23:22 -08:00
|
|
|
if code.type != "Text"
|
2018-11-09 16:40:36 -08:00
|
|
|
return code
|
2018-11-08 15:23:22 -08:00
|
|
|
operate_on_text = (text)->
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from(text.source)
|
2018-11-08 15:23:22 -08:00
|
|
|
for bit in *text
|
|
|
|
if type(bit) == "string"
|
|
|
|
lua\append bit
|
|
|
|
elseif bit.type == "Text"
|
|
|
|
lua\append(operate_on_text(bit))
|
|
|
|
else
|
|
|
|
lua\append compile(bit)
|
|
|
|
return lua
|
|
|
|
return operate_on_text code
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
["= lua"]: (compile, code)-> compile.action["lua >"](compile, code)
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
["use"]: (compile, path)-> LuaCode("run_file_1_in(#{compile(path)}, _ENV, OPTIMIZATION)")
|
|
|
|
|
|
|
|
["use 1 with prefix"]: (compile, path, prefix)->
|
|
|
|
LuaCode("run_file_1_in(#{compile(path)}, _ENV, OPTIMIZATION, ", compile(prefix), ")")
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
["tests"]: (compile)-> LuaCode("TESTS")
|
|
|
|
["test"]: (compile, body)->
|
2018-11-09 14:36:15 -08:00
|
|
|
unless body.type == 'Block'
|
2018-11-09 16:40:36 -08:00
|
|
|
compile_error(body, "This should be a Block")
|
2018-11-09 14:36:15 -08:00
|
|
|
test_nomsu = body\get_source_code!\match(":[ ]*(.*)")
|
|
|
|
if indent = test_nomsu\match("\n([ ]*)")
|
|
|
|
test_nomsu = test_nomsu\gsub("\n"..indent, "\n")
|
2018-11-09 16:40:36 -08:00
|
|
|
return LuaCode "TESTS[#{tostring(body.source)\as_lua!}] = ", test_nomsu\as_lua!
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
["is jit"]: (compile, code)-> LuaCode("jit")
|
|
|
|
["Lua version"]: (compile, code)-> LuaCode("_VERSION")
|
|
|
|
["nomsu environment"]: (compile)-> LuaCode("_ENV")
|
2018-11-08 15:23:22 -08:00
|
|
|
}
|
|
|
|
}, {
|
2018-11-17 14:38:05 -08:00
|
|
|
__import: import_to_1_from
|
|
|
|
__call: (compile, tree)->
|
2018-06-19 01:27:32 -07:00
|
|
|
switch tree.type
|
|
|
|
when "Action"
|
|
|
|
stub = tree.stub
|
2018-11-08 15:23:22 -08:00
|
|
|
compile_action = compile.action[stub]
|
2018-11-17 14:38:05 -08:00
|
|
|
if not compile_action and not tree.target and math_expression\match(stub)
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from(tree.source)
|
|
|
|
for i,tok in ipairs tree
|
|
|
|
if type(tok) == 'string'
|
|
|
|
lua\append tok
|
|
|
|
else
|
|
|
|
tok_lua = compile(tok)
|
|
|
|
tok_lua\parenthesize! if tok.type == "Action"
|
|
|
|
lua\append tok_lua
|
|
|
|
lua\append " " if i < #tree
|
|
|
|
return lua
|
|
|
|
|
2018-09-26 13:58:29 -07:00
|
|
|
if compile_action and not tree.target
|
2018-06-19 01:27:32 -07:00
|
|
|
args = [arg for arg in *tree when type(arg) != "string"]
|
|
|
|
-- Force Lua to avoid tail call optimization for debugging purposes
|
|
|
|
-- TODO: use tail call?
|
2018-11-09 16:40:36 -08:00
|
|
|
ret = compile_action(compile, unpack(args))
|
2018-09-16 17:52:59 -07:00
|
|
|
if ret == nil
|
2018-09-16 17:38:19 -07:00
|
|
|
info = debug.getinfo(compile_action, "S")
|
2018-09-16 17:52:59 -07:00
|
|
|
filename = Source\from_string(info.source).filename
|
2018-11-08 15:23:22 -08:00
|
|
|
compile_error tree,
|
2018-09-16 17:52:59 -07:00
|
|
|
"The compile-time action here (#{stub}) failed to return any value.",
|
|
|
|
"Look at the implementation of (#{stub}) in #{filename}:#{info.linedefined} and make sure it's returning something."
|
2018-11-06 15:13:55 -08:00
|
|
|
unless SyntaxTree\is_instance(ret)
|
2018-11-09 16:40:36 -08:00
|
|
|
ret.source or= tree.source
|
2018-11-06 15:13:55 -08:00
|
|
|
return ret
|
|
|
|
if ret != tree
|
2018-11-08 15:23:22 -08:00
|
|
|
return compile(ret)
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from(tree.source)
|
2018-09-10 15:55:34 -07:00
|
|
|
if tree.target -- Method call
|
2018-11-08 15:23:22 -08:00
|
|
|
target_lua = compile tree.target
|
|
|
|
target_text = target_lua\text!
|
2018-11-17 14:38:05 -08:00
|
|
|
-- TODO: this parenthesizing is maybe overly conservative
|
|
|
|
if target_text\match("^%(.*%)$") or target_text\match("^[_a-zA-Z][_a-zA-Z0-9.]*$") or
|
|
|
|
tree.target.type == "IndexChain"
|
2018-09-10 15:55:34 -07:00
|
|
|
lua\append target_lua, ":"
|
|
|
|
else
|
|
|
|
lua\append "(", target_lua, "):"
|
2018-09-10 16:36:51 -07:00
|
|
|
lua\append((stub)\as_lua_id!,"(")
|
2018-06-19 01:27:32 -07:00
|
|
|
args = {}
|
|
|
|
for i, tok in ipairs tree
|
|
|
|
if type(tok) == "string" then continue
|
2018-11-17 14:38:05 -08:00
|
|
|
arg_lua = compile(tok)
|
|
|
|
if tok.type == "Block"
|
|
|
|
arg_lua = LuaCode\from(tok.source, "(function()\n ", arg_lua, "\nend)()")
|
2018-06-19 01:27:32 -07:00
|
|
|
insert args, arg_lua
|
|
|
|
lua\concat_append args, ", "
|
|
|
|
lua\append ")"
|
|
|
|
return lua
|
|
|
|
|
|
|
|
when "EscapedNomsu"
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from tree.source, "SyntaxTree{"
|
2018-08-28 15:08:00 -07:00
|
|
|
needs_comma, i = false, 1
|
2018-09-18 19:48:58 -07:00
|
|
|
as_lua = (x)->
|
|
|
|
if type(x) == 'number'
|
|
|
|
tostring(x)
|
2018-10-31 15:54:18 -07:00
|
|
|
elseif SyntaxTree\is_instance(x)
|
2018-11-08 15:23:22 -08:00
|
|
|
compile(x)
|
2018-09-18 19:48:58 -07:00
|
|
|
else x\as_lua!
|
|
|
|
|
2018-10-31 15:54:18 -07:00
|
|
|
for k,v in pairs((SyntaxTree\is_instance(tree[1]) and tree[1].type == "EscapedNomsu" and tree) or tree[1])
|
2018-08-28 15:08:00 -07:00
|
|
|
if needs_comma then lua\append ", "
|
|
|
|
else needs_comma = true
|
|
|
|
if k == i
|
|
|
|
i += 1
|
|
|
|
elseif type(k) == 'string' and match(k,"[_a-zA-Z][_a-zA-Z0-9]*")
|
|
|
|
lua\append(k, "= ")
|
|
|
|
else
|
2018-09-18 19:48:58 -07:00
|
|
|
lua\append("[", as_lua(k), "]= ")
|
2018-08-28 15:08:00 -07:00
|
|
|
if k == "source"
|
2018-09-18 19:48:58 -07:00
|
|
|
lua\append tostring(v)\as_lua!
|
2018-08-28 15:08:00 -07:00
|
|
|
else
|
2018-09-18 19:48:58 -07:00
|
|
|
lua\append as_lua(v)
|
2018-08-28 15:08:00 -07:00
|
|
|
lua\append "}"
|
2018-08-27 13:38:58 -07:00
|
|
|
return lua
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
when "Block"
|
2018-11-17 14:38:05 -08:00
|
|
|
lua = LuaCode\from(tree.source)
|
|
|
|
for i, line in ipairs tree
|
|
|
|
if i > 1 then lua\append "\n"
|
|
|
|
lua\append compile(line)
|
|
|
|
return lua
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
when "Text"
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from(tree.source)
|
2018-11-26 16:21:42 -08:00
|
|
|
added = 0
|
2018-06-19 01:27:32 -07:00
|
|
|
string_buffer = ""
|
|
|
|
for i, bit in ipairs tree
|
|
|
|
if type(bit) == "string"
|
|
|
|
string_buffer ..= bit
|
|
|
|
continue
|
2018-09-18 19:48:58 -07:00
|
|
|
if string_buffer != ""
|
2018-11-26 16:21:42 -08:00
|
|
|
string_buffer = string_buffer\as_lua!
|
|
|
|
if lua\trailing_line_len! + #string_buffer > MAX_LINE
|
|
|
|
lua\append "\n "
|
|
|
|
if added > 0 then lua\append ".."
|
|
|
|
lua\append string_buffer
|
|
|
|
added += 1
|
2018-06-19 01:27:32 -07:00
|
|
|
string_buffer = ""
|
2018-11-08 15:23:22 -08:00
|
|
|
bit_lua = compile(bit)
|
2018-11-26 16:21:42 -08:00
|
|
|
if lua\trailing_line_len! + #bit_lua\text! > MAX_LINE
|
|
|
|
lua\append "\n "
|
|
|
|
if added > 0 then lua\append ".."
|
2018-06-19 01:27:32 -07:00
|
|
|
if bit.type != "Text"
|
2018-11-09 16:40:36 -08:00
|
|
|
bit_lua = LuaCode\from(bit.source, "tostring(",bit_lua,")")
|
2018-06-19 01:27:32 -07:00
|
|
|
lua\append bit_lua
|
2018-11-26 16:21:42 -08:00
|
|
|
added += 1
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
if string_buffer ~= "" or #lua.bits == 0
|
2018-11-26 16:21:42 -08:00
|
|
|
string_buffer = string_buffer\as_lua!
|
|
|
|
if lua\trailing_line_len! + #string_buffer > MAX_LINE
|
|
|
|
lua\append "\n "
|
|
|
|
if added > 0 then lua\append ".."
|
|
|
|
lua\append string_buffer
|
|
|
|
added += 1
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
if #lua.bits > 1
|
|
|
|
lua\parenthesize!
|
|
|
|
return lua
|
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
when "List", "Dict"
|
2018-11-17 14:38:05 -08:00
|
|
|
lua = LuaCode\from tree.source
|
2018-11-08 15:23:22 -08:00
|
|
|
i = 1
|
|
|
|
sep = ''
|
|
|
|
while i <= #tree
|
|
|
|
item = tree[i]
|
|
|
|
if item.type == "Block"
|
|
|
|
break
|
|
|
|
lua\append sep
|
|
|
|
if item.type == "Comment"
|
|
|
|
lua\append compile(item), "\n"
|
|
|
|
sep = ''
|
|
|
|
else
|
|
|
|
item_lua = compile(item)
|
|
|
|
lua\append item_lua
|
|
|
|
sep = ', '
|
|
|
|
i += 1
|
2018-11-17 14:38:05 -08:00
|
|
|
|
|
|
|
if lua\is_multiline!
|
|
|
|
lua = LuaCode\from tree.source, "#{tree.type}{\n ", lua, "\n}"
|
|
|
|
else
|
|
|
|
lua = LuaCode\from tree.source, "#{tree.type}{", lua, "}"
|
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
-- List/dict comprehenstion
|
2018-11-08 15:23:22 -08:00
|
|
|
if i <= #tree
|
2018-11-09 16:40:36 -08:00
|
|
|
lua = LuaCode\from tree.source, "(function()\n local comprehension = ", lua
|
|
|
|
if tree.type == "List"
|
|
|
|
lua\append "\n local function add(x) comprehension[#comprehension+1] = x end"
|
|
|
|
else
|
|
|
|
lua\append "\n local function #{("add 1 =")\as_lua_id!}(k, v) comprehension[k] = v end"
|
2018-11-08 15:23:22 -08:00
|
|
|
while i <= #tree
|
|
|
|
lua\append "\n "
|
|
|
|
if tree[i].type == 'Block' or tree[i].type == 'Comment'
|
|
|
|
lua\append compile(tree[i])
|
|
|
|
elseif tree[i].type == "DictEntry"
|
2018-11-09 16:40:36 -08:00
|
|
|
entry_lua = compile(tree[i])
|
|
|
|
lua\append (entry_lua\text!\sub(1,1) == '[' and "comprehension" or "comprehension."), entry_lua
|
2018-11-08 15:23:22 -08:00
|
|
|
else
|
2018-11-09 16:40:36 -08:00
|
|
|
lua\append "comprehension[#comprehension+1] = ", compile(tree[i])
|
2018-11-08 15:23:22 -08:00
|
|
|
i += 1
|
2018-11-09 16:40:36 -08:00
|
|
|
lua\append "\n return comprehension\nend)()"
|
2018-11-17 14:38:05 -08:00
|
|
|
|
2018-06-19 01:27:32 -07:00
|
|
|
return lua
|
|
|
|
|
|
|
|
when "DictEntry"
|
|
|
|
key, value = tree[1], tree[2]
|
2018-11-08 15:23:22 -08:00
|
|
|
key_lua = compile(key)
|
2018-11-09 16:40:36 -08:00
|
|
|
value_lua = value and compile(value) or LuaCode\from(key.source, "true")
|
2018-11-08 15:23:22 -08:00
|
|
|
key_str = match(key_lua\text!, [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=])
|
2018-09-26 13:58:29 -07:00
|
|
|
return if key_str and key_str\is_lua_id!
|
2018-11-09 16:40:36 -08:00
|
|
|
LuaCode\from tree.source, key_str,"=",value_lua
|
2018-11-08 15:23:22 -08:00
|
|
|
elseif sub(key_lua\text!,1,1) == "["
|
2018-06-19 01:27:32 -07:00
|
|
|
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
|
|
|
-- Lua's parser if the inner expression is a long string. Lua
|
|
|
|
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
2018-11-09 16:40:36 -08:00
|
|
|
LuaCode\from tree.source, "[ ",key_lua,"]=",value_lua
|
2018-06-19 01:27:32 -07:00
|
|
|
else
|
2018-11-09 16:40:36 -08:00
|
|
|
LuaCode\from tree.source, "[",key_lua,"]=",value_lua
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
when "IndexChain"
|
2018-11-08 15:23:22 -08:00
|
|
|
lua = compile(tree[1])
|
|
|
|
first_char = sub(lua\text!,1,1)
|
2018-06-19 01:27:32 -07:00
|
|
|
if first_char == "{" or first_char == '"' or first_char == "["
|
|
|
|
lua\parenthesize!
|
|
|
|
|
|
|
|
for i=2,#tree
|
|
|
|
key = tree[i]
|
2018-11-08 15:23:22 -08:00
|
|
|
key_lua = compile(key)
|
|
|
|
key_lua_str = key_lua\text!
|
2018-09-26 13:58:29 -07:00
|
|
|
lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
|
|
|
if lua_id and lua_id\is_lua_id!
|
2018-06-19 01:27:32 -07:00
|
|
|
lua\append ".#{lua_id}"
|
|
|
|
elseif sub(key_lua_str,1,1) == '['
|
|
|
|
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
|
|
|
-- Lua's parser if the inner expression is a long string. Lua
|
|
|
|
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
|
|
|
lua\append "[ ",key_lua," ]"
|
|
|
|
else
|
|
|
|
lua\append "[",key_lua,"]"
|
|
|
|
return lua
|
|
|
|
|
|
|
|
when "Number"
|
2018-11-09 16:40:36 -08:00
|
|
|
return LuaCode\from(tree.source, tostring(tree[1]))
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
when "Var"
|
2018-11-23 17:22:22 -08:00
|
|
|
return LuaCode\from(tree.source, (concat(tree, " "))\as_lua_id!)
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-07-15 19:41:22 -07:00
|
|
|
when "FileChunks"
|
2018-09-16 17:38:19 -07:00
|
|
|
error("Can't convert FileChunks to a single block of lua, since each chunk's "..
|
2018-06-19 01:27:32 -07:00
|
|
|
"compilation depends on the earlier chunks")
|
2018-09-12 15:31:59 -07:00
|
|
|
|
|
|
|
when "Comment"
|
2018-11-11 18:28:10 -08:00
|
|
|
return LuaCode\from(tree.source, "-- ", (tree[1]\gsub('\n', '\n-- ')))
|
2018-09-12 15:31:59 -07:00
|
|
|
|
|
|
|
when "Error"
|
2018-09-16 17:38:19 -07:00
|
|
|
error("Can't compile errors")
|
2018-06-19 01:27:32 -07:00
|
|
|
|
|
|
|
else
|
|
|
|
error("Unknown type: #{tree.type}")
|
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
})
|
2018-06-19 01:27:32 -07:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
return {:compile, :compile_error}
|