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
2019-01-29 16:16:45 -08:00
require ( " text " )
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 )
2019-02-05 15:45:27 -08:00
elseif not Source : is_instance ( source ) then
2019-01-29 16:16:45 -08:00
assert ( source.short_src and source.currentline )
file = Files.read ( source.short_src )
assert ( file , " Could not find " .. tostring ( source.short_src ) )
local lines = file : lines ( )
local start = 1
for i = 1 , source.currentline - 1 do
start = start + # lines [ i ]
end
local stop = start + # lines [ source.currentline ]
source = Source ( source.short_src , start , stop )
2018-11-17 14:38:05 -08:00
end
if source and not file then
file = Files.read ( source.filename )
end
2019-03-22 14:37:50 -07:00
if not file then
if NOMSU_PREFIX then
local path = tostring ( NOMSU_PREFIX ) .. " /share/nomsu/ " .. tostring ( table.concat ( NOMSU_VERSION , " . " ) ) .. " / " .. tostring ( source.filename )
file = Files.read ( path )
end
end
if not file then
error ( " Can't find file: " .. tostring ( 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
2019-01-27 02:02:40 -08:00
err_msg , hint = msg : match ( " (.*)[ \n ]+Hint:[ \n ]+(.*) " )
title = " Failure "
2019-01-16 16:31:49 -08:00
end
if not err_msg then
title , err_msg = msg : match ( " ([^:]*):[ \n ]+(.*) " )
end
if not err_msg then
err_msg = msg
2019-01-27 02:02:40 -08:00
title = " Failure "
2019-01-16 16:31:49 -08:00
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 )
2019-03-27 14:41:36 -07:00
if tree == nil then
error ( " No tree was passed in. " )
end
2019-03-09 15:57:01 -08:00
if tree.version and tree.version < self.NOMSU_VERSION : up_to ( # tree.version ) and self._1_upgraded_from_2_to then
tree = self._1_upgraded_from_2_to ( tree , tree.version , self.NOMSU_VERSION )
end
2019-01-10 16:33:37 -08:00
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
2019-02-05 15:45:27 -08:00
if not compile_action then
local seen_words = { }
local words = { }
for word in stub : gmatch ( " [^0-9 ][^ ]* " ) do
if not ( seen_words [ word ] ) then
seen_words [ word ] = true
table.insert ( words , word )
end
end
table.sort ( words )
local stub2 = table.concat ( words , " " )
compile_action = self.COMPILE_RULES [ stub2 ]
if compile_action then
if debug.getinfo ( compile_action , ' u ' ) . isvararg then
stub = stub2
else
compile_action = nil
end
end
end
2019-01-10 16:33:37 -08:00
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-18 14:22:17 -08:00
local ret = compile_action ( self , tree , unpack ( args ) )
2019-01-10 16:33:37 -08:00
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 )
2019-02-05 15:45:27 -08:00
if arg.type == " Block " and # arg > 1 then
2019-01-10 16:33:37 -08:00
arg_lua = LuaCode : from ( arg.source , " (function() \n " , arg_lua , " \n end)() " )
2018-11-08 15:23:22 -08:00
end
2019-03-09 15:57:01 -08:00
if lua : trailing_line_len ( ) + # arg_lua > MAX_LINE then
2019-01-10 16:33:37 -08:00
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
2019-01-25 15:49:29 -08:00
local stub = tree : get_stub ( )
local compile_action = self.COMPILE_RULES [ stub ]
if compile_action then
local args = tree : get_args ( )
local ret = compile_action ( self , tree , unpack ( args ) )
if ret == nil then
local info = debug.getinfo ( compile_action , " S " )
local filename = Source : from_string ( info.source ) . filename
fail_at ( tree , ( " Compile error: The compile-time method 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. " ) )
end
if not ( SyntaxTree : is_instance ( ret ) ) then
ret.source = ret.source or tree.source
return ret
end
if ret ~= tree then
return self : compile ( ret )
end
end
2019-01-10 16:33:37 -08:00
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
2019-02-05 15:45:27 -08:00
local self_lua = # tree > 2 and " _self " or target_lua
if # tree > 2 then
lua : add ( " (function( " , self_lua , " ) \n " )
end
2019-01-10 16:33:37 -08:00
for i = 2 , # tree do
if i > 2 then
2019-02-05 15:45:27 -08:00
lua : add ( " \n " )
2019-01-10 16:33:37 -08:00
end
2019-02-05 15:45:27 -08:00
if i > 2 and i == # tree then
lua : add ( " return " )
end
lua : add ( self_lua , " : " )
2019-01-10 16:33:37 -08:00
lua : add ( ( tree [ i ] . stub ) : as_lua_id ( ) , " ( " )
for argnum , arg in ipairs ( tree [ i ] : get_args ( ) ) do
local arg_lua = self : compile ( arg )
2019-02-05 15:45:27 -08:00
if arg.type == " Block " and # arg > 1 then
2018-12-14 17:49:36 -08:00
arg_lua = LuaCode : from ( arg.source , " (function() \n " , arg_lua , " \n end)() " )
end
2019-03-09 15:57:01 -08:00
if lua : trailing_line_len ( ) + # arg_lua > MAX_LINE then
2018-12-14 19:25:59 -08:00
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
2019-02-05 15:45:27 -08:00
if # tree > 2 then
lua : add ( " \n end)( " , target_lua , " ) " )
end
2019-01-10 16:33:37 -08:00
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-03-09 15:57:01 -08:00
if lua : trailing_line_len ( ) + # ( entry_lua : match ( " ^[ \n ]* " ) ) > MAX_LINE then
2019-01-10 16:33:37 -08:00
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 )
2019-03-27 14:41:36 -07:00
for i , line in ipairs ( tree ) do
if line.type == " Error " then
return self : compile ( line )
end
end
2019-01-10 16:33:37 -08:00
for i , line in ipairs ( tree ) do
if i > 1 then
lua : add ( " \n " )
2018-09-17 15:29:48 -07:00
end
2019-03-09 15:57:01 -08:00
local line_lua = self : compile ( line )
lua : add ( line_lua )
2019-03-13 20:53:30 -07:00
if not ( line_lua : last ( 1 ) == " ; " or line_lua : last ( 4 ) : match ( " [^_a-zA-Z0-9]end$ " ) ) then
2019-03-09 15:57:01 -08:00
lua : add ( " ; " )
end
2019-01-10 16:33:37 -08:00
end
return lua
elseif " Text " == _exp_0 then
2019-01-22 16:54:56 -08:00
if # tree == 0 then
return LuaCode : from ( tree.source , ' "" ' )
end
if # tree == 1 and type ( tree [ 1 ] ) == ' string ' then
return LuaCode : from ( tree.source , tree [ 1 ] : as_lua ( ) )
end
local lua = LuaCode : from ( tree.source , " Text( " )
2019-01-10 16:33:37 -08:00
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
2019-01-22 16:54:56 -08:00
lua : add ( " , \n " )
else
lua : add ( " , " )
2018-12-13 15:21:45 -08:00
end
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 " then
2019-01-22 16:15:25 -08:00
bit_lua = LuaCode : from ( bit.source , " a_List(function(add) " , " \n " , bit_lua , " \n end):joined() " )
2019-01-10 16:33:37 -08:00
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
2019-01-22 16:54:56 -08:00
return LuaCode : from ( tree.source , ' "" ' )
2019-01-10 16:33:37 -08:00
end
2019-01-22 16:54:56 -08:00
lua : add ( " ) " )
2019-01-10 16:33:37 -08:00
return lua
elseif " List " == _exp_0 or " Dict " == _exp_0 then
2019-01-22 16:15:25 -08:00
local typename = " a_ " .. tree.type
2019-01-10 16:33:37 -08:00
if # tree == 0 then
2019-01-22 16:15:25 -08:00
return LuaCode : from ( tree.source , typename , " {} " )
2019-01-10 16:33:37 -08:00
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
2019-01-22 16:15:25 -08:00
lua : add ( typename , " (function( " , ( tree.type == ' List ' and " add " or ( " add, " .. ( " add 1 = " ) : as_lua_id ( ) ) ) , " ) " )
2019-03-09 15:57:01 -08:00
local body = self : compile ( tree [ i ] )
body : declare_locals ( )
lua : add ( " \n " , body , " \n end) " )
2019-01-10 16:33:37 -08:00
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 ] )
2019-03-09 15:57:01 -08:00
if item_lua : match ( " ^%.[a-zA-Z_] " ) then
2019-01-10 16:33:37 -08:00
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
2019-03-09 15:57:01 -08:00
sep = items_lua : last ( 1 ) == " ; " and " \n " or " , \n "
2018-11-08 15:23:22 -08:00
else
2019-03-09 15:57:01 -08:00
sep = items_lua : last ( 1 ) == " ; " and " " or " , "
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
2019-01-22 16:15:25 -08:00
lua : add ( LuaCode : from ( items_lua.source , typename , " { \n " , items_lua , " \n } " ) )
2019-01-10 16:33:37 -08:00
else
2019-01-22 16:15:25 -08:00
lua : add ( LuaCode : from ( items_lua.source , typename , " { " , items_lua , " } " ) )
2019-01-10 16:33:37 -08:00
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 ] )
2019-03-09 15:57:01 -08:00
local key_str = key_lua : match ( ' ^"([a-zA-Z_][a-zA-Z0-9_]*)"$ ' )
2019-01-10 16:33:37 -08:00
if key_str and key_str : is_lua_id ( ) then
return LuaCode : from ( tree.source , " . " , key_str )
2019-03-09 15:57:01 -08:00
elseif key_lua : first ( 1 ) == " [ " then
2019-01-10 16:33:37 -08:00
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 ] )
2019-03-09 15:57:01 -08:00
if lua : match ( " [' \" }]$ " ) or lua : match ( " ]=*]$ " ) then
2019-01-10 16:33:37 -08:00
lua : parenthesize ( )
end
2019-02-05 15:45:27 -08:00
if lua : text ( ) == " ... " then
return LuaCode : from ( tree.source , " select( " , self : compile ( tree [ 2 ] [ 1 ] ) , " , ...) " )
end
2019-01-10 16:33:37 -08:00
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
2019-02-06 14:31:14 -08:00
local number = tostring ( tree [ 1 ] ) : gsub ( " _ " , " " )
return LuaCode : from ( tree.source , number )
2019-01-10 16:33:37 -08:00
elseif " Var " == _exp_0 then
2019-03-27 14:41:36 -07:00
if tree [ 1 ] . type == " MethodCall " then
return LuaCode : from ( tree.source , self : compile ( tree [ 1 ] [ 1 ] ) , " . " , tree [ 1 ] [ 2 ] : get_stub ( ) : as_lua_id ( ) )
end
2019-01-10 16:33:37 -08:00
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
2019-02-05 15:45:27 -08:00
local err_msg = pretty_error ( {
title = " Parse error " ,
error = tree.error ,
hint = tree.hint ,
source = tree : get_source_file ( ) ,
start = tree.source . start ,
stop = tree.source . stop ,
filename = tree.source . filename
} )
return error ( err_msg )
2019-01-10 16:33:37 -08:00
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
}