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-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
compile_error = function ( tree , err_msg , hint )
if hint == nil then
hint = nil
end
local err_str = pretty_error ( {
title = " Compile error " ,
error = err_msg ,
hint = hint ,
source = tree : get_source_code ( ) ,
start = tree.source . start ,
stop = tree.source . stop ,
filename = tree.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]+) " ")* [*/^+-] !. ]] )
local compile_math_expression
compile_math_expression = function ( compile , tree , ... )
local lua = LuaCode ( 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 ( )
2018-09-14 14:00:48 -07:00
end
2018-11-08 15:23:22 -08:00
lua : append ( tok_lua )
2018-09-14 14:00:48 -07:00
end
2018-11-08 15:23:22 -08:00
if i < # tree then
lua : append ( " " )
2018-09-14 14:00:48 -07:00
end
end
2018-11-08 15:23:22 -08:00
return lua
2018-09-12 15:31:59 -07:00
end
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 ( {
[ " " ] = function ( compile , tree , fn , ... )
local lua = LuaCode ( tree.source )
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-08 15:23:22 -08:00
lua : append ( compile ( select ( i , ... ) ) )
2018-11-06 15:13:55 -08:00
end
lua : append ( " ) " )
return lua
end ,
2018-11-08 15:23:22 -08:00
[ " Lua " ] = function ( compile , tree , code )
2018-09-06 12:46:39 -07:00
if code.type ~= " Text " then
2018-11-08 15:23:22 -08:00
return LuaCode ( code.source , " LuaCode( " , tostring ( code.source ) : as_lua ( ) , " , " , compile ( code ) , " ) " )
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 )
local lua = LuaCode ( text.source , " LuaCode( " , tostring ( text.source ) : as_lua ( ) )
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 ,
[ " Lua value " ] = function ( compile , tree , code )
return compile.action [ " Lua " ] ( compile , tree , code )
2018-06-19 01:27:32 -07:00
end ,
2018-11-08 15:23:22 -08:00
[ " lua > " ] = function ( compile , tree , code )
2018-09-06 12:46:39 -07:00
if code.type ~= " Text " then
2018-11-08 15:23:22 -08:00
return tree
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 )
local lua = LuaCode ( text.source )
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-08 15:23:22 -08:00
[ " = lua " ] = function ( compile , tree , code )
return compile.action [ " lua > " ] ( compile , tree , code )
2018-07-10 15:00:01 -07:00
end ,
2018-11-08 15:23:22 -08:00
[ " use " ] = function ( compile , tree , path )
return LuaCode ( tree.source , " run_file_1_in( " .. tostring ( compile ( path ) ) .. " , _ENV) " )
2018-07-30 14:26:08 -07:00
end ,
2018-11-08 15:23:22 -08:00
[ " tests " ] = function ( compile , tree )
return LuaCode ( tree.source , " TESTS " )
2018-07-30 14:26:08 -07:00
end ,
2018-11-08 15:23:22 -08:00
[ " test " ] = function ( compile , tree , body )
return LuaCode ( tree.source , " TESTS[ " .. tostring ( tostring ( tree.source ) : as_lua ( ) ) .. " ] = " , body : as_lua ( ) )
2018-09-16 16:57:14 -07:00
end ,
2018-11-08 15:23:22 -08:00
[ " is jit " ] = function ( compile , tree , code )
return LuaCode ( tree.source , " jit " )
end ,
[ " Lua version " ] = function ( compile , tree , code )
return LuaCode ( tree.source , " _VERSION " )
end ,
[ " nomsu environment " ] = function ( compile , tree )
return LuaCode ( tree.source , " _ENV " )
2018-06-19 01:27:32 -07:00
end
2018-11-08 15:23:22 -08:00
} )
} , {
__import = function ( self , other )
import_to_1_from ( self.action , other.action )
end ,
__call = function ( compile , tree , force_value )
2018-09-17 15:29:48 -07:00
if force_value == nil then
force_value = false
end
2018-07-09 19:22:40 -07:00
if tree.version then
do
2018-11-08 15:23:22 -08:00
local get_version = compile.action [ ( " Nomsu version " ) : as_lua_id ( ) ]
2018-07-17 23:08:13 -07:00
if get_version then
do
2018-11-08 15:23:22 -08:00
local upgrade = compile.action [ ( " 1 upgraded from 2 to " ) : as_lua_id ( ) ]
2018-07-17 23:08:13 -07:00
if upgrade then
tree = upgrade ( tree , tree.version , get_version ( ) )
end
end
2018-07-09 19:22:40 -07:00
end
end
end
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 ]
if not compile_action and math_expression : match ( stub ) then
compile_action = compile_math_expression
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-08 15:23:22 -08:00
local ret = compile_action ( compile , tree , 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
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-08 15:23:22 -08:00
local lua = LuaCode ( 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 ( )
if target_text : match ( " ^%(.*%)$ " ) or target_text : match ( " ^[_a-zA-Z][_a-zA-Z0-9.]*$ " ) 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-08 15:23:22 -08:00
local arg_lua = compile ( tok , true )
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-08 15:23:22 -08:00
local lua = LuaCode ( 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-09-17 15:29:48 -07:00
if not force_value then
local lua = LuaCode ( tree.source )
lua : concat_append ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # tree do
local line = tree [ _index_0 ]
2018-11-08 15:23:22 -08:00
_accum_0 [ _len_0 ] = compile ( line )
2018-09-17 15:29:48 -07:00
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " \n " )
return lua
else
2018-11-08 15:23:22 -08:00
local lua = LuaCode ( tree.source )
lua : append ( " ((function() " )
for i , line in ipairs ( tree ) do
lua : append ( " \n " , compile ( line ) )
2018-09-17 15:29:48 -07:00
end
2018-11-08 15:23:22 -08:00
lua : append ( " \n end)()) " )
2018-09-17 15:29:48 -07:00
return lua
end
2018-06-19 01:27:32 -07:00
elseif " Text " == _exp_0 then
2018-11-08 15:23:22 -08:00
local lua = LuaCode ( 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-08 15:23:22 -08:00
bit_lua = LuaCode ( 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
local lua = LuaCode ( tree.source , tostring ( tree.type ) .. " { " )
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-06-19 01:27:32 -07:00
lua : append ( " } " )
2018-11-08 15:23:22 -08:00
if i <= # tree then
lua = LuaCode ( tree.source , " (function() \n local it = " , lua )
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
lua : append ( " it[ " , compile ( tree [ i ] [ 1 ] ) , " ] = " , ( tree [ i ] [ 2 ] and compile ( tree [ i ] [ 2 ] ) or " true " ) )
else
lua : append ( " it:add( " , compile ( tree [ i ] ) , " ) " )
end
i = i + 1
end
lua : append ( " \n return it \n end)() " )
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 )
local value_lua = value and compile ( value ) or LuaCode ( key.source , " true " )
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-06-19 01:27:32 -07:00
return LuaCode ( tree.source , key_str , " = " , value_lua )
2018-11-08 15:23:22 -08:00
elseif sub ( key_lua : text ( ) , 1 , 1 ) == " [ " then
2018-06-19 01:27:32 -07:00
return LuaCode ( tree.source , " [ " , key_lua , " ]= " , value_lua )
else
return LuaCode ( tree.source , " [ " , key_lua , " ]= " , value_lua )
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-08 15:23:22 -08:00
return LuaCode ( tree.source , tostring ( tree [ 1 ] ) )
2018-06-19 01:27:32 -07:00
elseif " Var " == _exp_0 then
2018-11-08 15:23:22 -08:00
return LuaCode ( 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
return LuaCode ( tree.source , " " )
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
} )
return compile