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
[ " tests " ] = function ( compile )
return LuaCode ( " TESTS " )
2018-07-30 14:26:08 -07:00
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-11-17 14:38:05 -08:00
if not compile_action and not tree.target 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-09-26 13:58:29 -07:00
if compile_action and not tree.target then
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-08-29 14:19:16 -07:00
if tree.target then
2018-11-08 15:23:22 -08:00
local target_lua = compile ( tree.target )
local target_text = target_lua : text ( )
2018-11-17 14:38:05 -08:00
if target_text : match ( " ^%(.*%)$ " ) or target_text : match ( " ^[_a-zA-Z][_a-zA-Z0-9.]*$ " ) or tree.target . type == " IndexChain " then
2018-11-29 14:57:22 -08:00
lua : add ( target_lua , " : " )
2018-09-10 15:55:34 -07:00
else
2018-11-29 14:57:22 -08:00
lua : add ( " ( " , target_lua , " ): " )
2018-09-10 15:55:34 -07:00
end
2018-08-29 14:19:16 -07:00
end
2018-11-29 14:57:22 -08:00
lua : add ( ( stub ) : as_lua_id ( ) , " ( " )
2018-12-13 15:21:45 -08:00
local arg_count = 0
2018-06-19 01:27:32 -07:00
for i , tok in ipairs ( tree ) do
local _continue_0 = false
repeat
if type ( tok ) == " string " then
_continue_0 = true
break
end
2018-12-13 15:21:45 -08:00
arg_count = arg_count + 1
2018-11-17 14:38:05 -08:00
local arg_lua = compile ( tok )
if tok.type == " Block " then
arg_lua = LuaCode : from ( tok.source , " (function() \n " , arg_lua , " \n end)() " )
end
2018-12-13 15:21:45 -08:00
if arg_count > 1 then
lua : add ( " , " )
end
lua : add ( lua : trailing_line_len ( ) + # arg_lua : text ( ) > MAX_LINE and " \n " or " " )
lua : add ( arg_lua )
2018-06-19 01:27:32 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
end
end
2018-11-29 14:57:22 -08:00
lua : add ( " ) " )
2018-06-19 01:27:32 -07:00
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
lua : add ( " \n " )
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-11-29 14:23:02 -08:00
if bit.type == " Block " then
bit_lua = LuaCode : from ( bit.source , " (function() " , " \n local _buffer = List{} " , " \n local function add(bit) _buffer:add(bit) end " , " \n local function join_with(glue) _buffer = _buffer:joined_with(glue) end " , " \n " , bit_lua , " \n if lua_type_of(_buffer) == 'table' then _buffer = _buffer:joined() end " , " \n return _buffer " , " \n end)() " )
end
2018-06-19 01:27:32 -07:00
if bit.type ~= " Text " 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-11-17 14:38:05 -08:00
local lua = LuaCode : from ( tree.source )
2018-11-08 15:23:22 -08:00
local i = 1
local sep = ' '
while i <= # tree do
local item = tree [ i ]
if item.type == " Block " then
break
2018-06-19 01:27:32 -07:00
end
2018-11-29 14:57:22 -08:00
lua : add ( sep )
2018-11-08 15:23:22 -08:00
if item.type == " Comment " then
2018-11-29 14:57:22 -08:00
lua : add ( compile ( item ) , " \n " )
2018-11-08 15:23:22 -08:00
sep = ' '
else
local item_lua = compile ( item )
2018-11-29 14:57:22 -08:00
lua : add ( item_lua )
2018-11-08 15:23:22 -08:00
sep = ' , '
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
i = i + 1
end
2018-11-17 14:38:05 -08:00
if lua : is_multiline ( ) then
lua = LuaCode : from ( tree.source , tostring ( tree.type ) .. " { \n " , lua , " \n } " )
else
lua = LuaCode : from ( tree.source , tostring ( tree.type ) .. " { " , lua , " } " )
end
2018-11-08 15:23:22 -08:00
if i <= # tree then
2018-11-09 16:40:36 -08:00
lua = LuaCode : from ( tree.source , " (function() \n local comprehension = " , lua )
if tree.type == " List " then
2018-11-29 14:57:22 -08:00
lua : add ( " \n local function add(x) comprehension[#comprehension+1] = x end " )
2018-11-09 16:40:36 -08:00
else
2018-11-29 14:57:22 -08:00
lua : add ( " \n local function " .. tostring ( ( " add 1 = " ) : as_lua_id ( ) ) .. " (k, v) comprehension[k] = v end " )
2018-11-09 16:40:36 -08:00
end
2018-11-08 15:23:22 -08:00
while i <= # tree do
2018-11-29 14:57:22 -08:00
lua : add ( " \n " )
2018-11-08 15:23:22 -08:00
if tree [ i ] . type == ' Block ' or tree [ i ] . type == ' Comment ' then
2018-11-29 14:57:22 -08:00
lua : add ( compile ( tree [ i ] ) )
2018-11-08 15:23:22 -08:00
elseif tree [ i ] . type == " DictEntry " then
2018-11-09 16:40:36 -08:00
local entry_lua = compile ( tree [ i ] )
2018-11-29 14:57:22 -08:00
lua : add ( ( entry_lua : text ( ) : sub ( 1 , 1 ) == ' [ ' and " comprehension " or " comprehension. " ) , entry_lua )
2018-11-08 15:23:22 -08:00
else
2018-11-29 14:57:22 -08:00
lua : add ( " comprehension[#comprehension+1] = " , compile ( tree [ i ] ) )
2018-11-08 15:23:22 -08:00
end
i = i + 1
end
2018-11-29 14:57:22 -08:00
lua : add ( " \n return comprehension \n end)() " )
2018-11-08 15:23:22 -08:00
end
2018-06-19 01:27:32 -07:00
return lua
elseif " DictEntry " == _exp_0 then
local key , value = tree [ 1 ] , tree [ 2 ]
2018-11-08 15:23:22 -08:00
local key_lua = compile ( key )
2018-11-09 16:40:36 -08:00
local value_lua = value and compile ( value ) or LuaCode : from ( key.source , " true " )
2018-11-08 15:23:22 -08:00
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-11-09 16:40:36 -08:00
return LuaCode : from ( tree.source , key_str , " = " , value_lua )
2018-11-08 15:23:22 -08:00
elseif sub ( key_lua : text ( ) , 1 , 1 ) == " [ " then
2018-11-09 16:40:36 -08:00
return 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
return LuaCode : from ( tree.source , " [ " , key_lua , " ]= " , value_lua )
2018-06-19 01:27:32 -07:00
end
elseif " IndexChain " == _exp_0 then
2018-11-08 15:23:22 -08:00
local lua = compile ( tree [ 1 ] )
local first_char = sub ( lua : text ( ) , 1 , 1 )
2018-06-19 01:27:32 -07:00
if first_char == " { " or first_char == ' " ' or first_char == " [ " then
lua : parenthesize ( )
end
for i = 2 , # tree do
local key = tree [ i ]
2018-11-08 15:23:22 -08:00
local key_lua = compile ( key )
local key_lua_str = key_lua : text ( )
2018-09-26 13:58:29 -07:00
local lua_id = match ( key_lua_str , " ^[' \" ]([a-zA-Z_][a-zA-Z0-9_]*)[' \" ]$ " )
if lua_id and lua_id : is_lua_id ( ) then
2018-11-29 14:57:22 -08:00
lua : add ( " . " .. tostring ( lua_id ) )
2018-09-26 13:58:29 -07:00
elseif sub ( key_lua_str , 1 , 1 ) == ' [ ' then
2018-11-29 14:57:22 -08:00
lua : add ( " [ " , key_lua , " ] " )
2018-09-26 13:58:29 -07:00
else
2018-11-29 14:57:22 -08:00
lua : add ( " [ " , key_lua , " ] " )
2018-06-19 01:27:32 -07:00
end
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
2018-11-23 17:22:22 -08:00
return LuaCode : from ( tree.source , ( concat ( tree , " " ) ) : as_lua_id ( ) )
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
}