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-09 14:36:15 -08:00
local tree_to_nomsu , tree_to_inline_nomsu
do
local _obj_0 = require ( " nomsu_decompiler " )
tree_to_nomsu , tree_to_inline_nomsu = _obj_0.tree_to_nomsu , _obj_0.tree_to_inline_nomsu
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 )
lua : append ( fn_lua )
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
lua : append ( " ( " )
for i = 1 , select ( ' # ' , ... ) do
if i > 1 then
lua : append ( " , " )
end
2018-11-17 14:38:05 -08:00
lua : append ( compile ( ( select ( i , ... ) ) ) )
2018-11-06 15:13:55 -08:00
end
lua : append ( " ) " )
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 add_bit_lua
add_bit_lua = function ( lua , bit_lua )
local bit_leading_len = # ( bit_lua : match ( " ^[^ \n ]* " ) )
lua : append ( lua : trailing_line_len ( ) + bit_leading_len > MAX_LINE and " , \n " or " , " )
return lua : append ( bit_lua )
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 ]
if type ( bit ) == " string " then
add_bit_lua ( lua , bit : as_lua ( ) )
elseif bit.type == " Text " then
add_bit_lua ( lua , operate_on_text ( bit ) )
else
add_bit_lua ( lua , compile ( bit ) )
end
end
lua : append ( " ) " )
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
lua : append ( bit )
elseif bit.type == " Text " then
lua : append ( operate_on_text ( bit ) )
else
lua : append ( compile ( bit ) )
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-11-09 16:40:36 -08:00
return LuaCode ( " TESTS[ " .. tostring ( tostring ( body.source ) : as_lua ( ) ) .. " ] = " , test_nomsu : as_lua ( ) )
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
lua : append ( tok )
else
local tok_lua = compile ( tok )
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
lua : append ( tok_lua )
end
if i < # tree then
lua : append ( " " )
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-09-10 15:55:34 -07:00
lua : append ( target_lua , " : " )
else
lua : append ( " ( " , target_lua , " ): " )
end
2018-08-29 14:19:16 -07:00
end
2018-09-10 16:36:51 -07:00
lua : append ( ( stub ) : as_lua_id ( ) , " ( " )
2018-06-19 01:27:32 -07:00
local args = { }
for i , tok in ipairs ( tree ) do
local _continue_0 = false
repeat
if type ( tok ) == " string " then
_continue_0 = true
break
end
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-06-19 01:27:32 -07:00
insert ( args , arg_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
lua : concat_append ( args , " , " )
lua : append ( " ) " )
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-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-08-28 15:08:00 -07:00
if needs_comma then
lua : append ( " , " )
else
needs_comma = true
end
if k == i then
i = i + 1
elseif type ( k ) == ' string ' and match ( k , " [_a-zA-Z][_a-zA-Z0-9]* " ) then
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
end
if k == " source " then
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-06-19 01:27:32 -07:00
end
end
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
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
lua : append ( " \n " )
2018-09-17 15:29:48 -07:00
end
2018-11-17 14:38:05 -08:00
lua : append ( 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-06-19 01:27:32 -07:00
local string_buffer = " "
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
if # lua.bits > 0 then
lua : append ( " .. " )
end
2018-09-18 19:48:58 -07:00
lua : append ( string_buffer : as_lua ( ) )
2018-06-19 01:27:32 -07:00
string_buffer = " "
end
2018-11-08 15:23:22 -08:00
local bit_lua = compile ( bit )
2018-06-19 01:27:32 -07:00
if # lua.bits > 0 then
lua : append ( " .. " )
end
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
lua : append ( bit_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
if string_buffer ~= " " or # lua.bits == 0 then
if # lua.bits > 0 then
lua : append ( " .. " )
end
2018-09-18 19:48:58 -07:00
lua : append ( string_buffer : as_lua ( ) )
2018-06-19 01:27:32 -07:00
end
if # lua.bits > 1 then
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-08 15:23:22 -08:00
lua : append ( sep )
if item.type == " Comment " then
lua : append ( compile ( item ) , " \n " )
sep = ' '
else
local item_lua = compile ( item )
lua : append ( item_lua )
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
lua : append ( " \n local function add(x) comprehension[#comprehension+1] = x end " )
else
lua : append ( " \n local function " .. tostring ( ( " add 1 = " ) : as_lua_id ( ) ) .. " (k, v) comprehension[k] = v end " )
end
2018-11-08 15:23:22 -08:00
while i <= # tree do
lua : append ( " \n " )
if tree [ i ] . type == ' Block ' or tree [ i ] . type == ' Comment ' then
lua : append ( compile ( tree [ i ] ) )
elseif tree [ i ] . type == " DictEntry " then
2018-11-09 16:40:36 -08:00
local 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
end
i = i + 1
end
2018-11-09 16:40:36 -08:00
lua : append ( " \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
lua : append ( " . " .. tostring ( lua_id ) )
elseif sub ( key_lua_str , 1 , 1 ) == ' [ ' then
lua : append ( " [ " , key_lua , " ] " )
else
lua : append ( " [ " , 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-09 16:40:36 -08:00
return LuaCode : from ( tree.source , ( tree [ 1 ] ) : 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
}