2018-06-19 01:27:32 -07:00
local lpeg = require ( ' lpeg ' )
2018-09-14 14:38:59 -07:00
local R , P , S
R , P , S = lpeg.R , lpeg.P , lpeg.S
2018-06-19 01:27:32 -07:00
local re = require ( ' re ' )
2018-09-10 15:55:34 -07:00
local List , Dict , Text
2018-08-29 19:38:14 -07:00
do
local _obj_0 = require ( ' containers ' )
2018-09-10 15:55:34 -07:00
List , Dict , Text = _obj_0.List , _obj_0.Dict , _obj_0.Text
2018-08-29 19:38:14 -07:00
end
2018-06-19 01:27:32 -07:00
local insert , remove , concat
do
local _obj_0 = table
insert , remove , concat = _obj_0.insert , _obj_0.remove , _obj_0.concat
end
local unpack = unpack or table.unpack
2018-06-24 23:18:32 -07:00
local match , sub , gsub , format , byte , find
2018-06-19 01:27:32 -07:00
do
local _obj_0 = string
2018-06-24 23:18:32 -07:00
match , sub , gsub , format , byte , find = _obj_0.match , _obj_0.sub , _obj_0.gsub , _obj_0.format , _obj_0.byte , _obj_0.find
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
local LuaCode , Source
2018-06-19 01:27:32 -07:00
do
local _obj_0 = require ( " code_obj " )
2018-11-08 15:23:22 -08:00
LuaCode , Source = _obj_0.LuaCode , _obj_0.Source
2018-06-19 01:27:32 -07:00
end
2018-10-31 15:54:18 -07:00
local SyntaxTree = require ( " syntax_tree " )
2018-11-08 15:23:22 -08:00
local Importer , import_to_1_from , _1_forked
do
local _obj_0 = require ( ' importer ' )
Importer , import_to_1_from , _1_forked = _obj_0.Importer , _obj_0.import_to_1_from , _obj_0._1_forked
end
2018-11-17 14:38:05 -08:00
local Files = require ( " files " )
2018-08-31 15:21:47 -07:00
table.map = function ( t , fn )
return setmetatable ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _ , v in ipairs ( t ) do
_accum_0 [ _len_0 ] = fn ( v )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , getmetatable ( t ) )
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
local pretty_error = require ( " pretty_errors " )
local compile_error
2018-11-17 14:38:05 -08:00
compile_error = function ( source , err_msg , hint )
2018-11-08 15:23:22 -08:00
if hint == nil then
hint = nil
end
2018-11-17 14:38:05 -08:00
local file
if SyntaxTree : is_instance ( source ) then
file = source : get_source_file ( )
source = source.source
elseif type ( source ) == ' string ' then
source = Source : from_string ( source )
end
if source and not file then
file = Files.read ( source.filename )
end
2018-11-08 15:23:22 -08:00
local err_str = pretty_error ( {
title = " Compile error " ,
error = err_msg ,
hint = hint ,
2018-11-17 14:38:05 -08:00
source = file ,
start = source.start ,
stop = source.stop ,
filename = source.filename
2018-06-19 01:27:32 -07:00
} )
2018-11-08 15:23:22 -08:00
return error ( err_str , 0 )
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
local math_expression = re.compile ( [[ (([*/^+-] / [0-9]+) " ")* [*/^+-] !. ]] )
2018-06-19 01:27:32 -07:00
local MAX_LINE = 80
2018-11-08 15:23:22 -08:00
local compile = setmetatable ( {
action = Importer ( {
2018-11-09 16:40:36 -08:00
[ " " ] = function ( compile , fn , ... )
local lua = LuaCode ( )
2018-11-08 15:23:22 -08:00
local fn_lua = compile ( fn )
2018-11-29 14:57:22 -08:00
lua : add ( fn_lua )
2018-11-08 15:23:22 -08:00
if not ( fn_lua : text ( ) : match ( " ^%(.*%)$ " ) or fn_lua : text ( ) : match ( " ^[_a-zA-Z][_a-zA-Z0-9.]*$ " ) ) then
2018-11-06 15:13:55 -08:00
lua : parenthesize ( )
end
2018-11-29 14:57:22 -08:00
lua : add ( " ( " )
2018-11-06 15:13:55 -08:00
for i = 1 , select ( ' # ' , ... ) do
if i > 1 then
2018-11-29 14:57:22 -08:00
lua : add ( " , " )
2018-11-06 15:13:55 -08:00
end
2018-11-29 14:57:22 -08:00
lua : add ( compile ( ( select ( i , ... ) ) ) )
2018-11-06 15:13:55 -08:00
end
2018-11-29 14:57:22 -08:00
lua : add ( " ) " )
2018-11-06 15:13:55 -08:00
return lua
end ,
2018-11-09 16:40:36 -08:00
[ " Lua " ] = function ( compile , code )
2018-11-19 17:21:08 -08:00
if not code then
return LuaCode ( " LuaCode() " )
end
2018-09-06 12:46:39 -07:00
if code.type ~= " Text " then
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
end
local operate_on_text
operate_on_text = function ( text )
2018-11-09 16:40:36 -08:00
local lua = LuaCode : from ( text.source , " LuaCode:from( " , tostring ( text.source ) : as_lua ( ) )
2018-11-08 15:23:22 -08:00
for _index_0 = 1 , # text do
local bit = text [ _index_0 ]
2018-11-29 14:51:09 -08:00
local bit_lua
2018-11-08 15:23:22 -08:00
if type ( bit ) == " string " then
2018-11-29 14:51:09 -08:00
bit_lua = bit : as_lua ( )
2018-11-08 15:23:22 -08:00
elseif bit.type == " Text " then
2018-11-29 14:51:09 -08:00
bit_lua = operate_on_text ( bit )
elseif bit.type == " Block " then
2018-12-13 15:21:45 -08:00
bit_lua = LuaCode : from ( bit.source , " (function() " , " \n local _lua = LuaCode:from( " , tostring ( bit.source ) : as_lua ( ) , " ) " , " \n local function add(...) _lua:add(...) end " , " \n local function join_with(glue) " , " \n local old_bits = _lua.bits " , " \n _lua = LuaCode:from(_lua.source) " , " \n _lua:concat_add(old_bits, glue) " , " \n end " , " \n " , compile ( bit ) , " \n return _lua " , " \n end)() " )
2018-11-08 15:23:22 -08:00
else
2018-11-29 14:51:09 -08:00
bit_lua = compile ( bit )
2018-11-08 15:23:22 -08:00
end
2018-11-29 14:51:09 -08:00
local bit_leading_len = # ( bit_lua : match ( " ^[^ \n ]* " ) )
2018-11-29 14:57:22 -08:00
lua : add ( lua : trailing_line_len ( ) + bit_leading_len > MAX_LINE and " , \n " or " , " )
lua : add ( bit_lua )
2018-11-08 15:23:22 -08:00
end
2018-11-29 14:57:22 -08:00
lua : add ( " ) " )
2018-11-08 15:23:22 -08:00
return lua
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
return operate_on_text ( code )
end ,
2018-11-09 16:40:36 -08:00
[ " lua > " ] = function ( compile , code )
2018-09-06 12:46:39 -07:00
if code.type ~= " Text " then
2018-11-09 16:40:36 -08:00
return code
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
local operate_on_text
operate_on_text = function ( text )
2018-11-09 16:40:36 -08:00
local lua = LuaCode : from ( text.source )
2018-11-08 15:23:22 -08:00
for _index_0 = 1 , # text do
local bit = text [ _index_0 ]
if type ( bit ) == " string " then
2018-11-29 14:57:22 -08:00
lua : add ( bit )
2018-11-08 15:23:22 -08:00
elseif bit.type == " Text " then
2018-11-29 14:57:22 -08:00
lua : add ( operate_on_text ( bit ) )
2018-11-08 15:23:22 -08:00
else
2018-11-29 14:57:22 -08:00
lua : add ( compile ( bit ) )
2018-11-08 15:23:22 -08:00
end
2018-11-02 14:38:24 -07:00
end
2018-11-08 15:23:22 -08:00
return lua
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
return operate_on_text ( code )
2018-07-21 14:43:49 -07:00
end ,
2018-11-09 16:40:36 -08:00
[ " = lua " ] = function ( compile , code )
return compile.action [ " lua > " ] ( compile , code )
2018-07-10 15:00:01 -07:00
end ,
2018-11-09 16:40:36 -08:00
[ " use " ] = function ( compile , path )
2018-11-11 18:03:18 -08:00
return LuaCode ( " run_file_1_in( " .. tostring ( compile ( path ) ) .. " , _ENV, OPTIMIZATION) " )
2018-07-30 14:26:08 -07:00
end ,
2018-11-17 14:38:05 -08:00
[ " use 1 with prefix " ] = function ( compile , path , prefix )
return LuaCode ( " run_file_1_in( " .. tostring ( compile ( path ) ) .. " , _ENV, OPTIMIZATION, " , compile ( prefix ) , " ) " )
end ,
2018-11-09 16:40:36 -08:00
[ " test " ] = function ( compile , body )
2018-11-09 14:36:15 -08:00
if not ( body.type == ' Block ' ) then
2018-11-09 16:40:36 -08:00
compile_error ( body , " This should be a Block " )
2018-11-09 14:36:15 -08:00
end
local test_nomsu = body : get_source_code ( ) : match ( " :[ ]*(.*) " )
do
local indent = test_nomsu : match ( " \n ([ ]*) " )
if indent then
test_nomsu = test_nomsu : gsub ( " \n " .. indent , " \n " )
end
end
2018-12-13 15:21:45 -08:00
local test_text = compile ( SyntaxTree ( {
type = " Text " ,
source = body.source ,
test_nomsu
} ) )
return LuaCode ( " TESTS[ " .. tostring ( tostring ( body.source ) : as_lua ( ) ) .. " ] = " , test_text )
2018-09-16 16:57:14 -07:00
end ,
2018-11-09 16:40:36 -08:00
[ " is jit " ] = function ( compile , code )
return LuaCode ( " jit " )
2018-11-08 15:23:22 -08:00
end ,
2018-11-09 16:40:36 -08:00
[ " Lua version " ] = function ( compile , code )
return LuaCode ( " _VERSION " )
2018-11-08 15:23:22 -08:00
end ,
2018-11-09 16:40:36 -08:00
[ " nomsu environment " ] = function ( compile )
return LuaCode ( " _ENV " )
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
} )
} , {
2018-11-17 14:38:05 -08:00
__import = import_to_1_from ,
__call = function ( compile , tree )
2018-06-19 01:27:32 -07:00
local _exp_0 = tree.type
if " Action " == _exp_0 then
local stub = tree.stub
2018-11-08 15:23:22 -08:00
local compile_action = compile.action [ stub ]
2018-12-14 17:49:36 -08:00
if not compile_action and math_expression : match ( stub ) then
2018-11-09 16:40:36 -08:00
local lua = LuaCode : from ( tree.source )
for i , tok in ipairs ( tree ) do
if type ( tok ) == ' string ' then
2018-11-29 14:57:22 -08:00
lua : add ( tok )
2018-11-09 16:40:36 -08:00
else
local tok_lua = compile ( tok )
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
2018-11-29 14:57:22 -08:00
lua : add ( tok_lua )
2018-11-09 16:40:36 -08:00
end
if i < # tree then
2018-11-29 14:57:22 -08:00
lua : add ( " " )
2018-11-09 16:40:36 -08:00
end
end
return lua
2018-11-08 15:23:22 -08:00
end
2018-12-14 17:49:36 -08:00
if compile_action then
2018-09-26 13:58:29 -07:00
local args
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # tree do
local arg = tree [ _index_0 ]
if type ( arg ) ~= " string " then
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
2018-06-19 01:27:32 -07:00
end
end
2018-09-26 13:58:29 -07:00
args = _accum_0
end
2018-11-09 16:40:36 -08:00
local ret = compile_action ( compile , unpack ( args ) )
2018-09-26 13:58:29 -07:00
if ret == nil then
local info = debug.getinfo ( compile_action , " S " )
local filename = Source : from_string ( info.source ) . filename
2018-11-08 15:23:22 -08:00
compile_error ( tree , " The compile-time action here ( " .. tostring ( stub ) .. " ) failed to return any value. " , " Look at the implementation of ( " .. tostring ( stub ) .. " ) in " .. tostring ( filename ) .. " : " .. tostring ( info.linedefined ) .. " and make sure it's returning something. " )
2018-09-26 13:58:29 -07:00
end
2018-11-06 15:13:55 -08:00
if not ( SyntaxTree : is_instance ( ret ) ) then
2018-11-09 16:40:36 -08:00
ret.source = ret.source or tree.source
2018-11-06 15:13:55 -08:00
return ret
end
if ret ~= tree then
2018-11-08 15:23:22 -08:00
return compile ( ret )
2018-06-19 01:27:32 -07:00
end
end
2018-11-09 16:40:36 -08:00
local lua = LuaCode : from ( tree.source )
2018-12-14 17:49:36 -08:00
lua : add ( ( stub ) : as_lua_id ( ) , " ( " )
2018-12-14 19:25:59 -08:00
for argnum , arg in ipairs ( tree : get_args ( ) ) do
2018-12-14 17:49:36 -08:00
local arg_lua = compile ( arg )
if arg.type == " Block " then
arg_lua = LuaCode : from ( arg.source , " (function() \n " , arg_lua , " \n end)() " )
end
2018-12-14 19:25:59 -08:00
if lua : trailing_line_len ( ) + # arg_lua : text ( ) > MAX_LINE then
lua : add ( argnum > 1 and " , \n " or " \n " )
elseif argnum > 1 then
lua : add ( " , " )
2018-09-10 15:55:34 -07:00
end
2018-12-14 17:49:36 -08:00
lua : add ( arg_lua )
2018-08-29 14:19:16 -07:00
end
2018-12-14 17:49:36 -08:00
lua : add ( " ) " )
return lua
elseif " MethodCall " == _exp_0 then
local lua = LuaCode : from ( tree.source )
local target_lua = compile ( tree [ 1 ] )
local target_text = target_lua : text ( )
2018-12-14 19:23:26 -08:00
if not ( target_text : match ( " ^%(.*%)$ " ) or target_text : match ( " ^[_a-zA-Z][_a-zA-Z0-9.]*$ " ) or tree [ 1 ] . type == " IndexChain " ) then
target_lua : parenthesize ( )
2018-12-14 17:49:36 -08:00
end
2018-12-14 19:23:26 -08:00
for i = 2 , # tree do
if i > 2 then
lua : add ( " \n " )
2018-12-14 17:49:36 -08:00
end
2018-12-14 19:23:26 -08:00
lua : add ( target_lua , " : " )
lua : add ( ( tree [ i ] . stub ) : as_lua_id ( ) , " ( " )
for argnum , arg in ipairs ( tree [ i ] : get_args ( ) ) do
local arg_lua = compile ( arg )
if arg.type == " Block " then
arg_lua = LuaCode : from ( arg.source , " (function() \n " , arg_lua , " \n end)() " )
end
if lua : trailing_line_len ( ) + # arg_lua : text ( ) > MAX_LINE then
lua : add ( argnum > 1 and " , \n " or " \n " )
elseif argnum > 1 then
lua : add ( " , " )
end
lua : add ( arg_lua )
2018-06-19 01:27:32 -07:00
end
2018-12-14 19:23:26 -08:00
lua : add ( " ) " )
2018-06-19 01:27:32 -07:00
end
return lua
elseif " EscapedNomsu " == _exp_0 then
2018-11-09 16:40:36 -08:00
local lua = LuaCode : from ( tree.source , " SyntaxTree{ " )
2018-08-28 15:08:00 -07:00
local needs_comma , i = false , 1
2018-09-18 19:48:58 -07:00
local as_lua
as_lua = function ( x )
if type ( x ) == ' number ' then
return tostring ( x )
2018-10-31 15:54:18 -07:00
elseif SyntaxTree : is_instance ( x ) then
2018-11-08 15:23:22 -08:00
return compile ( x )
2018-12-13 15:21:45 -08:00
elseif Source : is_instance ( x ) then
return tostring ( x ) : as_lua ( )
2018-09-18 19:48:58 -07:00
else
return x : as_lua ( )
end
end
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 ] ) do
2018-12-13 15:21:45 -08:00
local entry_lua = LuaCode ( )
2018-08-28 15:08:00 -07:00
if k == i then
i = i + 1
elseif type ( k ) == ' string ' and match ( k , " [_a-zA-Z][_a-zA-Z0-9]* " ) then
2018-12-13 15:21:45 -08:00
entry_lua : add ( k , " = " )
2018-08-28 15:08:00 -07:00
else
2018-12-13 15:21:45 -08:00
entry_lua : add ( " [ " , as_lua ( k ) , " ]= " )
2018-08-28 15:08:00 -07:00
end
2018-12-13 15:21:45 -08:00
entry_lua : add ( as_lua ( v ) )
if needs_comma then
lua : add ( " , " )
2018-06-19 01:27:32 -07:00
end
2018-12-13 15:21:45 -08:00
if lua : trailing_line_len ( ) + # ( entry_lua : text ( ) : match ( " ^[ \n ]* " ) ) > MAX_LINE then
2019-01-01 15:05:58 -08:00
lua : add ( " \n " )
2018-12-13 15:21:45 -08:00
elseif needs_comma then
lua : add ( " " )
end
lua : add ( entry_lua )
needs_comma = true
2018-06-19 01:27:32 -07:00
end
2018-11-29 14:57:22 -08:00
lua : add ( " } " )
2018-08-27 13:38:58 -07:00
return lua
2018-06-19 01:27:32 -07:00
elseif " Block " == _exp_0 then
2018-11-17 14:38:05 -08:00
local lua = LuaCode : from ( tree.source )
for i , line in ipairs ( tree ) do
if i > 1 then
2018-11-29 14:57:22 -08:00
lua : add ( " \n " )
2018-09-17 15:29:48 -07:00
end
2018-11-29 14:57:22 -08:00
lua : add ( compile ( line ) )
2018-09-17 15:29:48 -07:00
end
2018-11-17 14:38:05 -08:00
return lua
2018-06-19 01:27:32 -07:00
elseif " Text " == _exp_0 then
2018-11-09 16:40:36 -08:00
local lua = LuaCode : from ( tree.source )
2018-11-26 16:21:42 -08:00
local added = 0
2018-06-19 01:27:32 -07:00
local string_buffer = " "
2018-12-13 15:21:45 -08:00
local add_bit
add_bit = function ( bit )
if added > 0 then
if lua : trailing_line_len ( ) + # bit > MAX_LINE then
lua : add ( " \n " )
end
lua : add ( " .. " )
end
lua : add ( bit )
added = added + 1
end
2018-06-19 01:27:32 -07:00
for i , bit in ipairs ( tree ) do
local _continue_0 = false
repeat
if type ( bit ) == " string " then
string_buffer = string_buffer .. bit
_continue_0 = true
break
end
if string_buffer ~= " " then
2018-12-13 15:21:45 -08:00
for i = 1 , # string_buffer , MAX_LINE do
add_bit ( string_buffer : sub ( i , i + MAX_LINE - 1 ) : as_lua ( ) )
2018-06-19 01:27:32 -07:00
end
string_buffer = " "
end
2018-11-08 15:23:22 -08:00
local bit_lua = compile ( bit )
2018-12-18 19:30:01 -08:00
if bit.type == " Block " and # bit == 1 then
bit = bit [ 1 ]
2018-11-29 14:23:02 -08:00
end
2018-12-18 19:30:01 -08:00
if bit.type == " Block " then
bit_lua = LuaCode : from ( bit.source , " List(function(add) " , " \n " , bit_lua , " \n end):joined() " )
elseif bit.type ~= " Text " and bit.type ~= " Number " then
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
end
2018-12-13 15:21:45 -08:00
add_bit ( bit_lua )
2018-06-19 01:27:32 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
end
end
2018-12-13 15:21:45 -08:00
if string_buffer ~= " " then
for i = 1 , # string_buffer , MAX_LINE do
add_bit ( string_buffer : sub ( i , i + MAX_LINE - 1 ) : as_lua ( ) )
2018-06-19 01:27:32 -07:00
end
2018-12-13 15:21:45 -08:00
string_buffer = " "
end
if added == 0 then
add_bit ( ' "" ' )
2018-06-19 01:27:32 -07:00
end
2018-12-13 15:21:45 -08:00
if added > 1 then
2018-06-19 01:27:32 -07:00
lua : parenthesize ( )
end
return lua
2018-11-08 15:23:22 -08:00
elseif " List " == _exp_0 or " Dict " == _exp_0 then
2018-12-18 19:30:01 -08:00
if # tree == 0 then
return LuaCode : from ( tree.source , tree.type , " {} " )
end
2018-11-17 14:38:05 -08:00
local lua = LuaCode : from ( tree.source )
2018-12-18 19:30:01 -08:00
local chunks = 0
2018-11-08 15:23:22 -08:00
local i = 1
2018-12-18 19:30:01 -08:00
while tree [ i ] do
if tree [ i ] . type == ' Block ' then
if chunks > 0 then
lua : add ( " + " )
end
lua : add ( tree.type , " (function( " , ( tree.type == ' List ' and " add " or ( " add, " .. ( " add 1 = " ) : as_lua_id ( ) ) ) , " ) " )
lua : add ( " \n " , compile ( tree [ i ] ) , " \n end) " )
chunks = chunks + 1
i = i + 1
2018-11-09 16:40:36 -08:00
else
2018-12-18 19:30:01 -08:00
if chunks > 0 then
lua : add ( " + " )
end
local sep = ' '
local items_lua = LuaCode : from ( tree [ i ] . source )
while tree [ i ] do
if tree [ i ] . type == " Block " then
break
end
local item_lua = compile ( tree [ i ] )
if item_lua : text ( ) : match ( " ^%.[a-zA-Z_] " ) then
item_lua = item_lua : text ( ) : sub ( 2 )
end
if tree.type == ' Dict ' and tree [ i ] . type == ' Index ' then
item_lua = LuaCode : from ( tree [ i ] . source , item_lua , " =true " )
end
items_lua : add ( sep , item_lua )
if tree [ i ] . type == " Comment " then
items_lua : add ( " \n " )
sep = ' '
2019-01-01 15:05:58 -08:00
elseif items_lua : trailing_line_len ( ) > MAX_LINE then
sep = ' , \n '
2018-12-18 19:30:01 -08:00
else
sep = ' , '
end
i = i + 1
end
if items_lua : is_multiline ( ) then
lua : add ( LuaCode : from ( items_lua.source , tree.type , " { \n " , items_lua , " \n } " ) )
2018-11-08 15:23:22 -08:00
else
2018-12-18 19:30:01 -08:00
lua : add ( LuaCode : from ( items_lua.source , tree.type , " { " , items_lua , " } " ) )
2018-11-08 15:23:22 -08:00
end
2018-12-18 19:30:01 -08:00
chunks = chunks + 1
2018-11-08 15:23:22 -08:00
end
end
2018-06-19 01:27:32 -07:00
return lua
2018-12-18 19:30:01 -08:00
elseif " Index " == _exp_0 then
local key_lua = compile ( tree [ 1 ] )
local key_str = match ( key_lua : text ( ) , ' ^"([a-zA-Z_][a-zA-Z0-9_]*)"$ ' )
2018-09-26 13:58:29 -07:00
if key_str and key_str : is_lua_id ( ) then
2018-12-18 19:30:01 -08:00
return LuaCode : from ( tree.source , " . " , key_str )
2018-11-08 15:23:22 -08:00
elseif sub ( key_lua : text ( ) , 1 , 1 ) == " [ " then
2018-12-18 19:30:01 -08:00
return LuaCode : from ( tree.source , " [ " , key_lua , " ] " )
2018-06-19 01:27:32 -07:00
else
2018-12-18 19:30:01 -08:00
return LuaCode : from ( tree.source , " [ " , key_lua , " ] " )
2018-06-19 01:27:32 -07:00
end
2018-12-18 19:30:01 -08:00
elseif " DictEntry " == _exp_0 then
local key = tree [ 1 ]
if key.type ~= " Index " then
key = SyntaxTree ( {
type = " Index " ,
source = key.source ,
key
} )
end
return LuaCode : from ( tree.source , compile ( key ) , " = " , ( tree [ 2 ] and compile ( tree [ 2 ] ) or " true " ) )
2018-06-19 01:27:32 -07:00
elseif " IndexChain " == _exp_0 then
2018-11-08 15:23:22 -08:00
local lua = compile ( tree [ 1 ] )
2018-12-18 19:30:01 -08:00
if lua : text ( ) : match ( " [' \" }]$ " ) or lua : text ( ) : match ( " ]=*]$ " ) then
2018-06-19 01:27:32 -07:00
lua : parenthesize ( )
end
for i = 2 , # tree do
local key = tree [ i ]
2018-12-18 19:30:01 -08:00
if key.type ~= " Index " then
key = SyntaxTree ( {
type = " Index " ,
source = key.source ,
key
} )
2018-06-19 01:27:32 -07:00
end
2018-12-18 19:30:01 -08:00
lua : add ( compile ( key ) )
2018-06-19 01:27:32 -07:00
end
return lua
elseif " Number " == _exp_0 then
2018-11-09 16:40:36 -08:00
return LuaCode : from ( tree.source , tostring ( tree [ 1 ] ) )
2018-06-19 01:27:32 -07:00
elseif " Var " == _exp_0 then
2019-01-01 15:52:56 -08:00
if type ( tree [ 1 ] ) == ' string ' then
return LuaCode : from ( tree.source , ( concat ( tree , " " ) ) : as_lua_id ( ) )
else
assert ( tree [ 1 ] . type == ' Action ' )
return LuaCode : from ( tree.source , tree [ 1 ] : get_stub ( ) : as_lua_id ( ) )
end
2018-07-15 19:41:22 -07:00
elseif " FileChunks " == _exp_0 then
2018-09-16 17:38:19 -07:00
return error ( " Can't convert FileChunks to a single block of lua, since each chunk's " .. " compilation depends on the earlier chunks " )
2018-09-12 15:31:59 -07:00
elseif " Comment " == _exp_0 then
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
elseif " Error " == _exp_0 then
2018-09-16 17:38:19 -07:00
return error ( " Can't compile errors " )
2018-06-19 01:27:32 -07:00
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
end
end
2018-11-08 15:23:22 -08:00
} )
2018-11-17 14:38:05 -08:00
return {
compile = compile ,
compile_error = compile_error
}