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 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-17 14:38:05 -08:00
local Files = require ( " files " )
2018-11-08 15:23:22 -08:00
local pretty_error = require ( " pretty_errors " )
2019-01-16 16:31:49 -08:00
local fail_at
fail_at = function ( source , msg )
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
2019-01-16 16:31:49 -08:00
local title , err_msg , hint = msg : match ( " ([^:]*):[ \n ]+(.*)[ \n ]+Hint: (.*) " )
if not err_msg then
err_msg , hint = msg : match ( " *(.*)[ \n ]+Hint:[ \n ]+(.*) " )
title = " Error "
end
if not err_msg then
title , err_msg = msg : match ( " ([^:]*):[ \n ]+(.*) " )
end
if not err_msg then
err_msg = msg
title = " Error "
end
2018-11-08 15:23:22 -08:00
local err_str = pretty_error ( {
2019-01-16 16:31:49 -08:00
title = title ,
2018-11-08 15:23:22 -08:00
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
2019-01-10 16:33:37 -08:00
local re = require ( ' re ' )
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
2019-01-10 16:33:37 -08:00
local compile
compile = function ( self , tree )
local _exp_0 = tree.type
if " Action " == _exp_0 then
local stub = tree.stub
local compile_action = self.COMPILE_RULES [ stub ]
if not compile_action and math_expression : match ( stub ) then
local lua = LuaCode : from ( tree.source )
for i , tok in ipairs ( tree ) do
if type ( tok ) == ' string ' then
lua : add ( tok )
else
local tok_lua = self : compile ( tok )
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
lua : add ( tok_lua )
end
if i < # tree then
lua : add ( " " )
2018-11-06 15:13:55 -08:00
end
end
return lua
2019-01-10 16:33:37 -08:00
end
if compile_action 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-11-08 15:23:22 -08:00
end
end
2019-01-10 16:33:37 -08:00
args = _accum_0
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
local ret = compile_action ( self , unpack ( args ) )
if ret == nil then
local info = debug.getinfo ( compile_action , " S " )
local filename = Source : from_string ( info.source ) . filename
2019-01-16 16:31:49 -08:00
fail_at ( tree , ( " Compile error: The compile-time action here ( " .. tostring ( stub ) .. " ) failed to return any value. " .. " Hint: Look at the implementation of ( " .. tostring ( stub ) .. " ) in " .. tostring ( filename ) .. " : " .. tostring ( info.linedefined ) .. " and make sure it's returning something. " ) )
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
if not ( SyntaxTree : is_instance ( ret ) ) then
ret.source = ret.source or tree.source
return ret
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
if ret ~= tree then
return self : compile ( ret )
2018-11-09 14:36:15 -08:00
end
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
local lua = LuaCode : from ( tree.source )
lua : add ( ( stub ) : as_lua_id ( ) , " ( " )
for argnum , arg in ipairs ( tree : get_args ( ) ) do
local arg_lua = self : compile ( arg )
if arg.type == " Block " then
arg_lua = LuaCode : from ( arg.source , " (function() \n " , arg_lua , " \n end)() " )
2018-11-08 15:23:22 -08:00
end
2019-01-10 16:33:37 -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-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
lua : add ( arg_lua )
end
lua : add ( " ) " )
return lua
elseif " MethodCall " == _exp_0 then
local lua = LuaCode : from ( tree.source )
local target_lua = self : compile ( tree [ 1 ] )
local target_text = target_lua : text ( )
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 ( )
end
for i = 2 , # tree do
if i > 2 then
lua : add ( " \n " )
end
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 = self : compile ( arg )
2018-12-14 17:49:36 -08:00
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 ( " ) " )
2019-01-10 16:33:37 -08:00
end
return lua
elseif " EscapedNomsu " == _exp_0 then
local lua = LuaCode : from ( tree.source , " SyntaxTree{ " )
local needs_comma , i = false , 1
local as_lua
as_lua = function ( x )
if type ( x ) == ' number ' then
return tostring ( x )
elseif SyntaxTree : is_instance ( x ) then
return self : compile ( x )
elseif Source : is_instance ( x ) then
return tostring ( x ) : as_lua ( )
else
return x : as_lua ( )
2018-12-14 17:49:36 -08:00
end
2019-01-10 16:33:37 -08:00
end
for k , v in pairs ( ( SyntaxTree : is_instance ( tree [ 1 ] ) and tree [ 1 ] . type == " EscapedNomsu " and tree ) or tree [ 1 ] ) do
local entry_lua = LuaCode ( )
if k == i then
i = i + 1
elseif type ( k ) == ' string ' and match ( k , " [_a-zA-Z][_a-zA-Z0-9]* " ) then
entry_lua : add ( k , " = " )
else
entry_lua : add ( " [ " , as_lua ( k ) , " ]= " )
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
entry_lua : add ( as_lua ( v ) )
if needs_comma then
lua : add ( " , " )
2018-09-18 19:48:58 -07:00
end
2019-01-10 16:33:37 -08:00
if lua : trailing_line_len ( ) + # ( entry_lua : text ( ) : match ( " ^[ \n ]* " ) ) > MAX_LINE then
lua : add ( " \n " )
elseif needs_comma then
lua : add ( " " )
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
lua : add ( entry_lua )
needs_comma = true
end
lua : add ( " } " )
return lua
elseif " Block " == _exp_0 then
local lua = LuaCode : from ( tree.source )
for i , line in ipairs ( tree ) do
if i > 1 then
lua : add ( " \n " )
2018-09-17 15:29:48 -07:00
end
2019-01-10 16:33:37 -08:00
lua : add ( self : compile ( line ) )
end
return lua
elseif " Text " == _exp_0 then
local lua = LuaCode : from ( tree.source )
local added = 0
local string_buffer = " "
local add_bit
add_bit = function ( bit )
if added > 0 then
if lua : trailing_line_len ( ) + # bit > MAX_LINE then
lua : add ( " \n " )
2018-12-13 15:21:45 -08:00
end
2019-01-10 16:33:37 -08:00
lua : add ( " .. " )
2018-12-13 15:21:45 -08:00
end
2019-01-10 16:33:37 -08:00
lua : add ( bit )
added = added + 1
end
for i , bit in ipairs ( tree ) do
local _continue_0 = false
repeat
if type ( bit ) == " string " then
string_buffer = string_buffer .. bit
2018-06-19 01:27:32 -07:00
_continue_0 = true
break
end
2019-01-10 16:33:37 -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 ( ) )
end
string_buffer = " "
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
local bit_lua = self : compile ( bit )
if bit.type == " Block " and # bit == 1 then
bit = bit [ 1 ]
end
if bit.type == " Block " then
bit_lua = LuaCode : from ( bit.source , " List(function(add) " , " \n " , bit_lua , " \n end):joined() " )
elseif bit.type ~= " Text " then
bit_lua = LuaCode : from ( bit.source , " tostring( " , bit_lua , " ) " )
end
add_bit ( bit_lua )
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
end
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-12-18 19:30:01 -08:00
end
2019-01-10 16:33:37 -08:00
string_buffer = " "
end
if added == 0 then
add_bit ( ' "" ' )
end
if added > 1 then
lua : parenthesize ( )
end
return lua
elseif " List " == _exp_0 or " Dict " == _exp_0 then
if # tree == 0 then
return LuaCode : from ( tree.source , tree.type , " {} " )
end
local lua = LuaCode : from ( tree.source )
local chunks = 0
local i = 1
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 " , self : compile ( tree [ i ] ) , " \n end) " )
chunks = chunks + 1
i = i + 1
else
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
2018-12-18 19:30:01 -08:00
end
2019-01-10 16:33:37 -08:00
local item_lua = self : compile ( tree [ i ] )
if item_lua : text ( ) : match ( " ^%.[a-zA-Z_] " ) then
item_lua = item_lua : text ( ) : sub ( 2 )
2018-12-18 19:30:01 -08:00
end
2019-01-10 16:33:37 -08:00
if tree.type == ' Dict ' and tree [ i ] . type == ' Index ' then
item_lua = LuaCode : from ( tree [ i ] . source , item_lua , " =true " )
2018-12-18 19:30:01 -08:00
end
2019-01-10 16:33:37 -08:00
items_lua : add ( sep , item_lua )
if tree [ i ] . type == " Comment " then
items_lua : add ( " \n " )
sep = ' '
elseif items_lua : trailing_line_len ( ) > MAX_LINE then
sep = ' , \n '
2018-11-08 15:23:22 -08:00
else
2019-01-10 16:33:37 -08:00
sep = ' , '
2018-11-08 15:23:22 -08:00
end
2019-01-10 16:33:37 -08:00
i = i + 1
2018-11-08 15:23:22 -08:00
end
2019-01-10 16:33:37 -08:00
if items_lua : is_multiline ( ) then
lua : add ( LuaCode : from ( items_lua.source , tree.type , " { \n " , items_lua , " \n } " ) )
else
lua : add ( LuaCode : from ( items_lua.source , tree.type , " { " , items_lua , " } " ) )
end
chunks = chunks + 1
2018-11-08 15:23:22 -08:00
end
2019-01-10 16:33:37 -08:00
end
return lua
elseif " Index " == _exp_0 then
local key_lua = self : compile ( tree [ 1 ] )
local key_str = match ( key_lua : text ( ) , ' ^"([a-zA-Z_][a-zA-Z0-9_]*)"$ ' )
if key_str and key_str : is_lua_id ( ) then
return LuaCode : from ( tree.source , " . " , key_str )
elseif sub ( key_lua : text ( ) , 1 , 1 ) == " [ " then
return LuaCode : from ( tree.source , " [ " , key_lua , " ] " )
else
return LuaCode : from ( tree.source , " [ " , key_lua , " ] " )
end
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 , self : compile ( key ) , " = " , ( tree [ 2 ] and self : compile ( tree [ 2 ] ) or " true " ) )
elseif " IndexChain " == _exp_0 then
local lua = self : compile ( tree [ 1 ] )
if lua : text ( ) : match ( " [' \" }]$ " ) or lua : text ( ) : match ( " ]=*]$ " ) then
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
} )
end
2019-01-10 16:33:37 -08:00
lua : add ( self : compile ( key ) )
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
return lua
elseif " Number " == _exp_0 then
return LuaCode : from ( tree.source , tostring ( tree [ 1 ] ) )
elseif " Var " == _exp_0 then
return LuaCode : from ( tree.source , tree : as_var ( ) : as_lua_id ( ) )
elseif " FileChunks " == _exp_0 then
return error ( " Can't convert FileChunks to a single block of lua, since each chunk's " .. " compilation depends on the earlier chunks " )
elseif " Comment " == _exp_0 then
return LuaCode : from ( tree.source , " -- " , ( tree [ 1 ] : gsub ( ' \n ' , ' \n -- ' ) ) )
elseif " Error " == _exp_0 then
return error ( " Can't compile errors " )
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
2018-06-19 01:27:32 -07:00
end
2019-01-10 16:33:37 -08:00
end
2018-11-17 14:38:05 -08:00
return {
compile = compile ,
2019-01-16 16:31:49 -08:00
fail_at = fail_at
2018-11-17 14:38:05 -08:00
}