2018-06-19 01:27:32 -07:00
local lpeg = require ( ' lpeg ' )
local re = require ( ' re ' )
local utils = require ( ' utils ' )
2018-07-21 14:43:49 -07:00
local Files = require ( ' files ' )
2018-06-19 01:27:32 -07:00
local repr , stringify , equivalent
repr , stringify , equivalent = utils.repr , utils.stringify , utils.equivalent
colors = require ( ' consolecolors ' )
colored = setmetatable ( { } , {
__index = function ( _ , color )
return ( function ( msg )
return colors [ color ] .. tostring ( msg or ' ' ) .. colors.reset
end )
end
} )
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
local NomsuCode , LuaCode , Source
do
local _obj_0 = require ( " code_obj " )
NomsuCode , LuaCode , Source = _obj_0.NomsuCode , _obj_0.LuaCode , _obj_0.Source
end
2018-07-23 15:28:35 -07:00
local AST = require ( " syntax_tree " )
2018-06-23 00:57:31 -07:00
local Parser = require ( " parser " )
2018-06-19 01:27:32 -07:00
SOURCE_MAP = { }
string.as_lua_id = function ( str )
str = gsub ( str , " x([0-9A-F][0-9A-F]) " , " x \0 %1 " )
str = gsub ( str , " %W " , function ( c )
if c == ' ' then
return ' _ '
else
return format ( " x%02X " , byte ( c ) )
end
end )
2018-08-29 14:19:16 -07:00
return str
2018-06-19 01:27:32 -07:00
end
table.map = function ( self , fn )
local _accum_0 = { }
local _len_0 = 1
for _ , v in ipairs ( self ) do
_accum_0 [ _len_0 ] = fn ( v )
_len_0 = _len_0 + 1
end
return _accum_0
end
table.fork = function ( t , values )
return setmetatable ( values or { } , {
__index = t
} )
end
2018-08-28 15:08:00 -07:00
table.copy = function ( t )
return setmetatable ( ( function ( )
local _tbl_0 = { }
for k , v in pairs ( t ) do
_tbl_0 [ k ] = v
end
return _tbl_0
end ) ( ) , getmetatable ( t ) )
end
2018-06-19 01:27:32 -07:00
do
local STRING_METATABLE = getmetatable ( " " )
STRING_METATABLE.__add = function ( self , other )
return self .. stringify ( other )
end
STRING_METATABLE.__index = function ( self , i )
local ret = string [ i ]
if ret ~= nil then
return ret
end
if type ( i ) == ' number ' then
return sub ( self , i , i )
elseif type ( i ) == ' table ' then
return sub ( self , i [ 1 ] , i [ 2 ] )
end
end
end
local _list_mt = {
__eq = equivalent ,
__tostring = function ( self )
return " [ " .. concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # self do
local b = self [ _index_0 ]
_accum_0 [ _len_0 ] = repr ( b )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " , " ) .. " ] "
2018-06-23 00:57:31 -07:00
end ,
__lt = function ( self , other )
assert ( type ( self ) == ' table ' and type ( other ) == ' table ' , " Incompatible types for comparison " )
for i = 1 , math.max ( # self , # other ) do
2018-07-17 17:25:12 -07:00
if not self [ i ] and other [ i ] then
return true
elseif self [ i ] and not other [ i ] then
return false
elseif self [ i ] < other [ i ] then
2018-06-23 00:57:31 -07:00
return true
elseif self [ i ] > other [ i ] then
return false
end
end
return false
end ,
__le = function ( self , other )
assert ( type ( self ) == ' table ' and type ( other ) == ' table ' , " Incompatible types for comparison " )
for i = 1 , math.max ( # self , # other ) do
2018-07-17 17:25:12 -07:00
if not self [ i ] and other [ i ] then
return true
elseif self [ i ] and not other [ i ] then
return false
elseif self [ i ] < other [ i ] then
2018-06-23 00:57:31 -07:00
return true
elseif self [ i ] > other [ i ] then
return false
end
end
return true
2018-08-29 15:59:30 -07:00
end ,
__index = {
add_1 = insert ,
append_1 = insert ,
add_1_at_index_2 = function ( t , x , i )
return insert ( t , i , x )
end ,
at_index_1_add_2 = insert ,
pop = table.remove ,
remove_last = table.remove ,
remove_index_1 = table.remove
}
2018-06-19 01:27:32 -07:00
}
local list
list = function ( t )
return setmetatable ( t , _list_mt )
end
local _dict_mt = {
__eq = equivalent ,
__tostring = function ( self )
return " { " .. concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for k , v in pairs ( self ) do
_accum_0 [ _len_0 ] = tostring ( repr ( k ) ) .. " : " .. tostring ( repr ( v ) )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " , " ) .. " } "
end
}
local dict
dict = function ( t )
return setmetatable ( t , _dict_mt )
end
local MAX_LINE = 80
2018-08-27 13:38:58 -07:00
local NomsuCompiler = setmetatable ( {
name = " Nomsu "
} , {
2018-06-19 01:27:32 -07:00
__index = function ( self , k )
do
local _self = rawget ( self , " self " )
if _self then
return _self [ k ]
else
return nil
end
end
2018-08-27 13:38:58 -07:00
end ,
__tostring = function ( self )
return self.name
2018-06-19 01:27:32 -07:00
end
} )
do
2018-07-24 15:08:44 -07:00
NomsuCompiler.NOMSU_COMPILER_VERSION = 5
2018-06-23 00:57:31 -07:00
NomsuCompiler.NOMSU_SYNTAX_VERSION = Parser.version
2018-08-28 15:08:00 -07:00
NomsuCompiler.nomsu = NomsuCompiler
2018-06-19 01:27:32 -07:00
NomsuCompiler.parse = function ( self , ... )
2018-06-23 00:57:31 -07:00
return Parser.parse ( ... )
end
NomsuCompiler.can_optimize = function ( )
return false
2018-06-19 01:27:32 -07:00
end
local to_add = {
repr = repr ,
stringify = stringify ,
utils = utils ,
lpeg = lpeg ,
re = re ,
2018-07-21 14:43:49 -07:00
Files = Files ,
2018-06-19 01:27:32 -07:00
next = next ,
unpack = unpack ,
setmetatable = setmetatable ,
coroutine = coroutine ,
rawequal = rawequal ,
getmetatable = getmetatable ,
pcall = pcall ,
error = error ,
package = package ,
os = os ,
require = require ,
tonumber = tonumber ,
tostring = tostring ,
string = string ,
xpcall = xpcall ,
module = module ,
print = print ,
loadfile = loadfile ,
rawset = rawset ,
_VERSION = _VERSION ,
collectgarbage = collectgarbage ,
rawget = rawget ,
rawlen = rawlen ,
table = table ,
assert = assert ,
dofile = dofile ,
loadstring = loadstring ,
type = type ,
select = select ,
math = math ,
io = io ,
load = load ,
pairs = pairs ,
ipairs = ipairs ,
list = list ,
dict = dict
}
2018-07-30 14:26:08 -07:00
if jit then
to_add.bit = require ( ' bit ' )
elseif _VERSION == " Lua 5.2 " then
to_add.bit = bit32
end
2018-06-19 01:27:32 -07:00
for k , v in pairs ( to_add ) do
NomsuCompiler [ k ] = v
end
for k , v in pairs ( AST ) do
NomsuCompiler [ k ] = v
end
NomsuCompiler.LuaCode = LuaCode
NomsuCompiler.NomsuCode = NomsuCode
NomsuCompiler.Source = Source
NomsuCompiler.LOADED = { }
2018-07-21 14:43:49 -07:00
NomsuCompiler.TESTS = { }
2018-06-19 01:27:32 -07:00
NomsuCompiler.AST = AST
2018-07-18 17:55:29 -07:00
NomsuCompiler.compile_error = function ( self , source , err_format_string , ... )
err_format_string = err_format_string : gsub ( " %%[^s] " , " %%%1 " )
2018-07-21 14:43:49 -07:00
local file = Files.read ( source.filename )
local line_starts = Files.get_line_starts ( file )
local line_no = Files.get_line_number ( file , source.start )
2018-06-24 23:18:32 -07:00
local line_start = line_starts [ line_no ]
2018-07-18 17:55:29 -07:00
local src = colored.dim ( file : sub ( line_start , source.start - 1 ) )
src = src .. colored.underscore ( colored.bright ( colored.red ( file : sub ( source.start , source.stop - 1 ) ) ) )
2018-07-21 14:43:49 -07:00
local end_of_line = ( line_starts [ Files.get_line_number ( file , source.stop ) + 1 ] or 0 ) - 1
2018-07-18 17:55:29 -07:00
src = src .. colored.dim ( file : sub ( source.stop , end_of_line - 1 ) )
2018-06-19 01:27:32 -07:00
src = ' ' .. src : gsub ( ' \n ' , ' \n ' )
local err_msg = err_format_string : format ( src , ... )
2018-07-18 17:55:29 -07:00
return error ( tostring ( source.filename ) .. " : " .. tostring ( line_no ) .. " : " .. err_msg , 0 )
2018-06-19 01:27:32 -07:00
end
2018-08-28 15:08:00 -07:00
local math_expression = re.compile ( [[ ([+-] " ")* [0-9]+ (" " [*/^+-] (" " [+-])* " " [0-9]+)+ !. ]] )
2018-06-19 01:27:32 -07:00
local add_lua_bits
2018-07-10 17:10:37 -07:00
add_lua_bits = function ( self , val_or_stmt , code )
local cls = val_or_stmt == " value " and LuaCode.Value or LuaCode
local operate_on_text
operate_on_text = function ( text )
local lua = cls ( 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
local bit_lua = self : compile ( bit )
if not ( bit_lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( bit.source , " Cannot use: \n %s \n as a string interpolation value, since it's not an expression. " )
2018-07-10 17:10:37 -07:00
end
lua : append ( bit_lua )
2018-06-19 01:27:32 -07:00
end
end
2018-07-10 17:10:37 -07:00
return lua
2018-06-19 01:27:32 -07:00
end
2018-07-10 17:10:37 -07:00
return operate_on_text ( code )
2018-06-19 01:27:32 -07:00
end
local add_lua_string_bits
2018-07-10 17:10:37 -07:00
add_lua_string_bits = function ( self , val_or_stmt , code )
local cls_str = val_or_stmt == " value " and " LuaCode.Value( " or " LuaCode( "
2018-06-19 01:27:32 -07:00
if code.type ~= " Text " then
2018-07-10 17:36:52 -07:00
return LuaCode.Value ( code.source , cls_str , repr ( tostring ( code.source ) ) , " , " , self : compile ( code ) , " ) " )
2018-06-19 01:27:32 -07:00
end
2018-07-10 17:10:37 -07:00
local add_bit_lua
add_bit_lua = function ( lua , bit_lua )
2018-08-28 15:08:00 -07:00
local bit_leading_len = # ( bit_lua : match ( " ^[^ \n ]* " ) )
2018-07-17 16:13:35 -07:00
lua : append ( lua : trailing_line_len ( ) + bit_leading_len > MAX_LINE and " , \n " or " , " )
2018-07-10 17:10:37 -07:00
return lua : append ( bit_lua )
end
local operate_on_text
operate_on_text = function ( text )
local lua = LuaCode.Value ( text.source , cls_str , repr ( tostring ( text.source ) ) )
for _index_0 = 1 , # text do
local bit = text [ _index_0 ]
if type ( bit ) == " string " then
add_bit_lua ( lua , repr ( bit ) )
elseif bit.type == " Text " then
add_bit_lua ( lua , operate_on_text ( bit ) )
else
local bit_lua = self : compile ( bit )
if not ( bit_lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( bit.source , " Cannot use: \n %s \n as a string interpolation value, since it's not an expression. " )
2018-07-10 17:10:37 -07:00
end
add_bit_lua ( lua , bit_lua )
end
end
lua : append ( " ) " )
return lua
2018-06-19 01:27:32 -07:00
end
2018-07-10 17:10:37 -07:00
return operate_on_text ( code )
2018-06-19 01:27:32 -07:00
end
NomsuCompiler.COMPILE_ACTIONS = setmetatable ( {
[ " # compile math expr # " ] = function ( self , tree , ... )
local lua = LuaCode.Value ( tree.source )
for i , tok in ipairs ( tree ) do
if type ( tok ) == ' string ' then
lua : append ( tok )
else
local tok_lua = self : compile ( tok )
if not ( tok_lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( tok.source , " Non-expression value inside math expression: \n %s " )
2018-06-19 01:27:32 -07:00
end
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
lua : append ( tok_lua )
end
if i < # tree then
lua : append ( " " )
end
end
return lua
end ,
2018-08-28 15:08:00 -07:00
[ " Lua 1 " ] = function ( self , tree , _code )
2018-07-10 17:10:37 -07:00
return add_lua_string_bits ( self , ' statements ' , _code )
2018-06-19 01:27:32 -07:00
end ,
2018-08-28 15:08:00 -07:00
[ " Lua value 1 " ] = function ( self , tree , _code )
2018-07-10 17:10:37 -07:00
return add_lua_string_bits ( self , ' value ' , _code )
2018-06-19 01:27:32 -07:00
end ,
2018-08-28 15:08:00 -07:00
[ " lua > 1 " ] = function ( self , tree , _code )
2018-06-19 01:27:32 -07:00
if _code.type ~= " Text " then
2018-08-28 15:08:00 -07:00
return LuaCode ( tree.source , " nomsu:run_lua( " , self : compile ( _code ) , " ); " )
2018-06-19 01:27:32 -07:00
end
2018-07-10 17:10:37 -07:00
return add_lua_bits ( self , " statements " , _code )
2018-06-19 01:27:32 -07:00
end ,
2018-08-28 15:08:00 -07:00
[ " = lua 1 " ] = function ( self , tree , _code )
2018-06-19 01:27:32 -07:00
if _code.type ~= " Text " then
2018-08-28 15:08:00 -07:00
return LuaCode.Value ( tree.source , " nomsu:run_lua( " , self : compile ( _code ) , " :as_statements('return ')) " )
2018-06-19 01:27:32 -07:00
end
2018-07-10 17:10:37 -07:00
return add_lua_bits ( self , " value " , _code )
2018-06-19 01:27:32 -07:00
end ,
2018-08-28 15:08:00 -07:00
[ " use 1 " ] = function ( self , tree , _path )
2018-06-24 16:11:08 -07:00
if _path.type == ' Text ' and # _path == 1 and type ( _path [ 1 ] ) == ' string ' then
local path = _path [ 1 ]
2018-07-21 14:43:49 -07:00
for _ , f in Files.walk ( path ) do
2018-06-24 16:11:08 -07:00
self : run_file ( f )
end
2018-06-19 01:27:32 -07:00
end
2018-08-28 15:08:00 -07:00
return LuaCode ( tree.source , " for i,f in Files.walk( " , self : compile ( _path ) , " ) do nomsu:run_file(f) end " )
2018-07-21 14:43:49 -07:00
end ,
[ " tests " ] = function ( self , tree )
return LuaCode.Value ( tree.source , " TESTS " )
2018-07-10 15:00:01 -07:00
end ,
2018-08-28 15:08:00 -07:00
[ " test 1 " ] = function ( self , tree , _body )
2018-07-21 14:43:49 -07:00
local test_str = table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # _body do
local line = _body [ _index_0 ]
_accum_0 [ _len_0 ] = tostring ( self : tree_to_nomsu ( line ) )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " \n " )
return LuaCode ( tree.source , " TESTS[ " .. tostring ( repr ( tostring ( tree.source ) ) ) .. " ] = " , repr ( test_str ) )
2018-07-30 14:26:08 -07:00
end ,
[ " is jit " ] = function ( self , tree , _code )
return LuaCode.Value ( tree.source , jit and " true " or " false " )
end ,
[ " Lua version " ] = function ( self , tree , _code )
return LuaCode.Value ( tree.source , repr ( _VERSION ) )
2018-06-19 01:27:32 -07:00
end
} , {
__index = function ( self , stub )
if math_expression : match ( stub ) then
return self [ " # compile math expr # " ]
end
end
} )
2018-07-15 19:41:22 -07:00
NomsuCompiler.run = function ( self , to_run , source , version )
2018-06-19 01:27:32 -07:00
if source == nil then
source = nil
end
2018-07-15 19:41:22 -07:00
if version == nil then
version = nil
end
2018-06-23 00:57:31 -07:00
source = source or ( to_run.source or Source ( to_run , 1 , # to_run ) )
2018-07-13 14:30:32 -07:00
if type ( source ) == ' string ' then
source = Source : from_string ( source )
end
2018-07-21 14:43:49 -07:00
if not Files.read ( source.filename ) then
Files.spoof ( source.filename , to_run )
2018-06-23 00:57:31 -07:00
end
2018-06-19 01:27:32 -07:00
local tree
if AST.is_syntax_tree ( to_run ) then
2018-06-23 00:57:31 -07:00
tree = to_run
2018-06-19 01:27:32 -07:00
else
2018-07-15 19:41:22 -07:00
tree = self : parse ( to_run , source , version )
2018-06-19 01:27:32 -07:00
end
if tree == nil then
return nil
end
2018-07-15 19:41:22 -07:00
if tree.type ~= " FileChunks " then
tree = {
tree
}
end
local ret = nil
local all_lua = { }
for _index_0 = 1 , # tree do
local chunk = tree [ _index_0 ]
local lua = self : compile ( chunk ) : as_statements ( " return " )
2018-06-19 01:27:32 -07:00
lua : declare_locals ( )
2018-06-23 00:57:31 -07:00
lua : prepend ( " -- File: " .. tostring ( source.filename : gsub ( " \n .* " , " ... " ) ) .. " \n " )
2018-07-15 19:41:22 -07:00
insert ( all_lua , tostring ( lua ) )
ret = self : run_lua ( lua )
2018-06-19 01:27:32 -07:00
end
2018-07-15 19:41:22 -07:00
return ret
2018-06-19 01:27:32 -07:00
end
local _running_files = { }
NomsuCompiler.run_file = function ( self , filename )
if self.LOADED [ filename ] then
return self.LOADED [ filename ]
end
2018-06-24 16:11:08 -07:00
for i , running in ipairs ( _running_files ) do
if running == filename then
local loop
2018-06-19 01:27:32 -07:00
do
2018-06-24 16:11:08 -07:00
local _accum_0 = { }
local _len_0 = 1
for j = i , # _running_files do
_accum_0 [ _len_0 ] = _running_files [ j ]
_len_0 = _len_0 + 1
2018-06-19 01:27:32 -07:00
end
2018-06-24 16:11:08 -07:00
loop = _accum_0
2018-06-19 01:27:32 -07:00
end
2018-06-24 16:11:08 -07:00
insert ( loop , filename )
error ( " Circular import, this loops forever: " .. tostring ( concat ( loop , " -> " ) ) .. " ... " )
end
end
insert ( _running_files , filename )
local ret = nil
if match ( filename , " %.lua$ " ) then
2018-07-21 14:43:49 -07:00
local file = assert ( Files.read ( filename ) , " Could not find file: " .. tostring ( filename ) )
2018-06-24 16:11:08 -07:00
ret = self : run_lua ( file , Source ( filename , 1 , # file ) )
elseif match ( filename , " %.nom$ " ) or match ( filename , " ^/dev/fd/[012]$ " ) then
local ran_lua
if self.can_optimize ( filename ) then
local lua_filename = gsub ( filename , " %.nom$ " , " .lua " )
do
2018-07-21 14:43:49 -07:00
local file = Files.read ( lua_filename )
2018-06-24 16:11:08 -07:00
if file then
ret = self : run_lua ( file , Source ( lua_filename , 1 , # file ) )
ran_lua = true
2018-06-19 01:27:32 -07:00
end
end
2018-06-24 16:11:08 -07:00
end
if not ( ran_lua ) then
2018-07-21 14:43:49 -07:00
local file = Files.read ( filename )
2018-06-24 16:11:08 -07:00
if not file then
2018-07-24 15:08:44 -07:00
error ( " Tried to run file that does not exist: " .. tostring ( filename ) )
2018-06-19 01:27:32 -07:00
end
2018-06-24 16:11:08 -07:00
ret = self : run ( file , Source ( filename , 1 , # file ) )
2018-06-19 01:27:32 -07:00
end
2018-06-24 16:11:08 -07:00
else
error ( " Invalid filetype for " .. tostring ( filename ) , 0 )
2018-06-19 01:27:32 -07:00
end
self.LOADED [ filename ] = ret or true
2018-06-24 16:11:08 -07:00
remove ( _running_files )
self.LOADED [ filename ] = ret or true
2018-06-19 01:27:32 -07:00
return ret
end
NomsuCompiler.run_lua = function ( self , lua , source )
if source == nil then
source = nil
end
local lua_string = tostring ( lua )
2018-08-28 15:08:00 -07:00
local run_lua_fn , err = load ( lua_string , nil and tostring ( source or lua.source ) , " t " , self )
2018-06-19 01:27:32 -07:00
if not run_lua_fn then
local line_numbered_lua = concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
2018-07-21 14:43:49 -07:00
for i , line in ipairs ( Files.get_lines ( lua_string ) ) do
2018-06-19 01:27:32 -07:00
_accum_0 [ _len_0 ] = format ( " %3d|%s " , i , line )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " \n " )
error ( " Failed to compile generated code: \n " .. tostring ( colored.bright ( colored.blue ( colored.onblack ( line_numbered_lua ) ) ) ) .. " \n \n " .. tostring ( err ) , 0 )
end
2018-07-10 15:00:01 -07:00
source = source or lua.source
local source_key = tostring ( source )
2018-06-19 01:27:32 -07:00
if not ( SOURCE_MAP [ source_key ] ) then
local map = { }
2018-07-21 14:43:49 -07:00
local file = Files.read ( source.filename )
2018-06-23 17:22:23 -07:00
if not file then
error ( " Failed to find file: " .. tostring ( source.filename ) )
end
2018-08-28 15:08:00 -07:00
local nomsu_str = file : sub ( source.start , source.stop )
assert ( type ( nomsu_str ) == ' string ' )
2018-06-19 01:27:32 -07:00
local lua_line = 1
2018-07-21 14:43:49 -07:00
local nomsu_line = Files.get_line_number ( file , source.start )
2018-07-10 15:00:01 -07:00
local map_sources
map_sources = function ( s )
2018-06-19 01:27:32 -07:00
if type ( s ) == ' string ' then
for nl in s : gmatch ( " \n " ) do
map [ lua_line ] = map [ lua_line ] or nomsu_line
lua_line = lua_line + 1
end
else
2018-07-10 15:00:01 -07:00
if s.source and s.source . filename == source.filename then
2018-07-21 14:43:49 -07:00
nomsu_line = Files.get_line_number ( file , s.source . start )
2018-06-19 01:27:32 -07:00
end
local _list_0 = s.bits
for _index_0 = 1 , # _list_0 do
local b = _list_0 [ _index_0 ]
2018-07-10 15:00:01 -07:00
map_sources ( b )
2018-06-19 01:27:32 -07:00
end
end
end
2018-07-10 15:00:01 -07:00
map_sources ( lua )
2018-07-10 17:13:48 -07:00
map [ lua_line ] = map [ lua_line ] or nomsu_line
2018-06-19 01:27:32 -07:00
map [ 0 ] = 0
SOURCE_MAP [ source_key ] = map
end
return run_lua_fn ( )
end
NomsuCompiler.compile = function ( self , tree )
2018-07-09 19:22:40 -07:00
if tree.version then
do
2018-08-29 14:19:16 -07:00
local get_version = self [ ' A_ ' .. string.as_lua_id ( " Nomsu version " ) ]
2018-07-17 23:08:13 -07:00
if get_version then
do
2018-08-29 14:19:16 -07:00
local upgrade = self [ ' A_ ' .. string.as_lua_id ( " 1 upgraded from 2 to 3 " ) ]
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
do
local compile_action = self.COMPILE_ACTIONS [ stub ]
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
end
end
args = _accum_0
end
local ret = compile_action ( self , tree , unpack ( args ) )
if not ret then
2018-07-18 17:55:29 -07:00
self : compile_error ( tree.source , " Compile-time action: \n %s \n failed to produce any Lua " )
2018-06-19 01:27:32 -07:00
end
return ret
end
end
2018-08-29 14:19:16 -07:00
local lua = LuaCode.Value ( tree.source )
if tree.target then
lua : append ( self : compile ( tree.target ) , " : " )
else
lua : append ( " A_ " )
end
lua : append ( string.as_lua_id ( stub ) , " ( " )
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
local arg_lua = self : compile ( tok )
if not ( arg_lua.is_value ) then
2018-08-27 13:38:58 -07:00
if tok.type == " Block " then
self : compile_error ( tok.source , ( " Cannot compile action ' " .. tostring ( stub ) .. " ' with a Block as an argument. \n " .. " Maybe there should be a compile-time action with that name that isn't being found? " ) )
else
self : compile_error ( tok.source , " Cannot use: \n %s \n as an argument to '%s', since it's not an expression, it produces: %s " , stub , repr ( arg_lua ) )
end
2018-06-19 01:27:32 -07:00
end
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-08-28 15:08:00 -07:00
local lua = LuaCode.Value ( tree.source , tree [ 1 ] . type , " { " )
local needs_comma , i = false , 1
for k , v in pairs ( AST.is_syntax_tree ( tree [ 1 ] , " EscapedNomsu " ) and tree or tree [ 1 ] ) do
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
lua : append ( " [ " , ( AST.is_syntax_tree ( k ) and self : compile ( k ) or repr ( k ) ) , " ]= " )
end
if k == " source " then
lua : append ( repr ( tostring ( v ) ) )
else
lua : append ( AST.is_syntax_tree ( v ) and self : compile ( v ) or repr ( 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
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 ]
_accum_0 [ _len_0 ] = self : compile ( line ) : as_statements ( )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " \n " )
return lua
elseif " Text " == _exp_0 then
local lua = LuaCode.Value ( tree.source )
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
lua : append ( repr ( string_buffer ) )
string_buffer = " "
end
local bit_lua = self : compile ( bit )
if not ( bit_lua.is_value ) then
2018-08-27 13:38:58 -07:00
local src = ' ' .. gsub ( tostring ( self : compile ( bit ) ) , ' \n ' , ' \n ' )
2018-07-21 14:43:49 -07:00
local line = tostring ( bit.source . filename ) .. " : " .. tostring ( Files.get_line_number ( Files.read ( bit.source . filename ) , bit.source . start ) )
2018-07-18 17:55:29 -07:00
self : compile_error ( bit.source , " Cannot use: \n %s \n as a string interpolation value, since it's not an expression. " )
2018-06-19 01:27:32 -07:00
end
if # lua.bits > 0 then
lua : append ( " .. " )
end
if bit.type ~= " Text " then
bit_lua = LuaCode.Value ( bit.source , " stringify( " , bit_lua , " ) " )
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
lua : append ( repr ( string_buffer ) )
end
if # lua.bits > 1 then
lua : parenthesize ( )
end
return lua
elseif " List " == _exp_0 then
local lua = LuaCode.Value ( tree.source , " list{ " )
2018-07-22 15:59:29 -07:00
lua : concat_append ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # tree do
local e = tree [ _index_0 ]
_accum_0 [ _len_0 ] = self : compile ( e )
_len_0 = _len_0 + 1
2018-06-19 01:27:32 -07:00
end
2018-07-22 15:59:29 -07:00
return _accum_0
end ) ( ) , " , " , " , \n " )
2018-06-19 01:27:32 -07:00
lua : append ( " } " )
return lua
elseif " Dict " == _exp_0 then
local lua = LuaCode.Value ( tree.source , " dict{ " )
lua : concat_append ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # tree do
local e = tree [ _index_0 ]
_accum_0 [ _len_0 ] = self : compile ( e )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " , " , " , \n " )
lua : append ( " } " )
return lua
elseif " DictEntry " == _exp_0 then
local key , value = tree [ 1 ] , tree [ 2 ]
local key_lua = self : compile ( key )
if not ( key_lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( tree [ 1 ] . source , " Cannot use: \n %s \n as a dict key, since it's not an expression. " )
2018-06-19 01:27:32 -07:00
end
local value_lua = value and self : compile ( value ) or LuaCode.Value ( key.source , " true " )
if not ( value_lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( tree [ 2 ] . source , " Cannot use: \n %s \n as a dict value, since it's not an expression. " )
2018-06-19 01:27:32 -07:00
end
2018-08-28 15:08:00 -07:00
local key_str = match ( tostring ( key_lua ) , [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=] )
2018-06-19 01:27:32 -07:00
if key_str then
return LuaCode ( tree.source , key_str , " = " , value_lua )
elseif sub ( tostring ( key_lua ) , 1 , 1 ) == " [ " then
return LuaCode ( tree.source , " [ " , key_lua , " ]= " , value_lua )
else
return LuaCode ( tree.source , " [ " , key_lua , " ]= " , value_lua )
end
elseif " IndexChain " == _exp_0 then
local lua = self : compile ( tree [ 1 ] )
if not ( lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( tree [ 1 ] . source , " Cannot index: \n %s \n since it's not an expression. " )
2018-06-19 01:27:32 -07:00
end
local first_char = sub ( tostring ( lua ) , 1 , 1 )
if first_char == " { " or first_char == ' " ' or first_char == " [ " then
lua : parenthesize ( )
end
for i = 2 , # tree do
local key = tree [ i ]
local key_lua = self : compile ( key )
if not ( key_lua.is_value ) then
2018-07-18 17:55:29 -07:00
self : compile_error ( key.source , " Cannot use: \n %s \n as an index, since it's not an expression. " )
2018-06-19 01:27:32 -07:00
end
local key_lua_str = tostring ( key_lua )
do
local lua_id = match ( key_lua_str , " ^[' \" ]([a-zA-Z_][a-zA-Z0-9_]*)[' \" ]$ " )
if 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 , " ] " )
end
end
end
return lua
elseif " Number " == _exp_0 then
return LuaCode.Value ( tree.source , tostring ( tree [ 1 ] ) )
elseif " Var " == _exp_0 then
2018-08-29 14:19:16 -07:00
return LuaCode.Value ( tree.source , " _ " , string.as_lua_id ( tree [ 1 ] ) )
2018-07-15 19:41:22 -07:00
elseif " FileChunks " == _exp_0 then
return error ( " Cannot convert FileChunks to a single block of lua, since each chunk's " .. " compilation depends on the earlier chunks " )
2018-06-19 01:27:32 -07:00
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
end
end
2018-07-20 17:51:21 -07:00
NomsuCompiler.tree_to_inline_nomsu = function ( self , tree , parenthesize_blocks , check , len )
2018-07-19 20:41:31 -07:00
if parenthesize_blocks == nil then
parenthesize_blocks = false
end
2018-07-20 17:51:21 -07:00
if check == nil then
check = nil
end
if len == nil then
len = 0
end
local recurse
recurse = function ( tree , nomsu , parenthesize_blocks )
if nomsu == nil then
nomsu = nil
end
if parenthesize_blocks == nil then
parenthesize_blocks = false
end
return self : tree_to_inline_nomsu ( tree , parenthesize_blocks , check , len + ( nomsu and # tostring ( nomsu ) or 0 ) )
end
2018-07-19 20:41:31 -07:00
local _exp_0 = tree.type
if " FileChunks " == _exp_0 then
return error ( " Cannot inline a FileChunks " )
elseif " Action " == _exp_0 then
local nomsu = NomsuCode ( tree.source )
2018-08-28 15:08:00 -07:00
if tree.target then
nomsu : append ( self : tree_to_inline_nomsu ( tree.target ) , " :: " )
end
2018-07-19 20:41:31 -07:00
for i , bit in ipairs ( tree ) do
if type ( bit ) == " string " then
local clump_words = ( type ( tree [ i - 1 ] ) == ' string ' and Parser.is_operator ( bit ) ~= Parser.is_operator ( tree [ i - 1 ] ) )
if i > 1 and not clump_words then
nomsu : append ( " " )
end
nomsu : append ( bit )
else
2018-07-20 17:51:21 -07:00
local arg_nomsu = recurse ( bit , nomsu , parenthesize_blocks or ( i == 1 or i < # tree ) )
2018-08-28 15:08:00 -07:00
if not ( arg_nomsu : match ( " ^: " ) or i == 1 ) then
2018-07-19 20:41:31 -07:00
nomsu : append ( " " )
end
if bit.type == " Action " then
arg_nomsu : parenthesize ( )
end
nomsu : append ( arg_nomsu )
end
2018-07-20 17:51:21 -07:00
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-20 17:51:21 -07:00
end
2018-07-19 20:41:31 -07:00
end
return nomsu
elseif " EscapedNomsu " == _exp_0 then
2018-07-20 17:51:21 -07:00
local inner_nomsu = recurse ( tree [ 1 ] )
if not ( tree [ 1 ] . type == " List " or tree [ 1 ] . type == " Dict " or tree [ 1 ] . type == " Var " ) then
inner_nomsu : parenthesize ( )
end
local nomsu = NomsuCode ( tree.source , " \\ " , inner_nomsu )
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-19 20:41:31 -07:00
end
2018-07-20 17:51:21 -07:00
return nomsu
2018-07-19 20:41:31 -07:00
elseif " Block " == _exp_0 then
2018-07-20 19:36:05 -07:00
local nomsu = NomsuCode ( tree.source , " : " )
2018-07-20 17:51:21 -07:00
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-20 17:51:21 -07:00
end
2018-07-19 20:41:31 -07:00
for i , line in ipairs ( tree ) do
nomsu : append ( i == 1 and " " or " ; " )
2018-07-20 17:51:21 -07:00
nomsu : append ( recurse ( line , nomsu , i == 1 or i < # tree ) )
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-20 17:51:21 -07:00
end
2018-07-19 20:41:31 -07:00
end
if # tree > 1 or parenthesize_blocks then
nomsu : parenthesize ( )
end
return nomsu
elseif " Text " == _exp_0 then
2018-07-20 20:13:01 -07:00
local add_text
add_text = function ( nomsu , tree )
2018-07-19 20:41:31 -07:00
for i , bit in ipairs ( tree ) do
if type ( bit ) == ' string ' then
2018-07-20 20:13:01 -07:00
local escaped = Parser.inline_escape ( bit )
nomsu : append ( Parser.inline_escape ( bit ) )
2018-07-19 20:41:31 -07:00
elseif bit.type == " Text " then
2018-07-20 20:13:01 -07:00
add_text ( nomsu , bit )
2018-07-19 20:41:31 -07:00
else
2018-07-20 17:51:21 -07:00
local interp_nomsu = recurse ( bit , nomsu )
2018-07-19 20:41:31 -07:00
if bit.type ~= " Var " and bit.type ~= " List " and bit.type ~= " Dict " then
interp_nomsu : parenthesize ( )
elseif bit.type == " Var " and type ( tree [ i + 1 ] ) == ' string ' and not match ( tree [ i + 1 ] , " ^[ \n \t ,.:;#(){}[%]] " ) then
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
end
2018-07-20 17:51:21 -07:00
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-20 17:51:21 -07:00
end
2018-07-19 20:41:31 -07:00
end
end
2018-07-20 20:13:01 -07:00
local nomsu = NomsuCode ( tree.source )
add_text ( nomsu , tree )
return NomsuCode ( tree.source , ' " ' , nomsu , ' " ' )
2018-07-22 15:59:29 -07:00
elseif " List " == _exp_0 or " Dict " == _exp_0 then
local nomsu = NomsuCode ( tree.source , ( tree.type == " List " and " [ " or " { " ) )
2018-07-19 20:41:31 -07:00
for i , item in ipairs ( tree ) do
if i > 1 then
nomsu : append ( " , " )
end
2018-07-20 17:51:21 -07:00
nomsu : append ( recurse ( item , nomsu ) )
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-20 17:51:21 -07:00
end
2018-07-19 20:41:31 -07:00
end
2018-07-22 15:59:29 -07:00
nomsu : append ( tree.type == " List " and " ] " or " } " )
2018-07-19 20:41:31 -07:00
return nomsu
elseif " DictEntry " == _exp_0 then
local key , value = tree [ 1 ] , tree [ 2 ]
2018-07-20 17:51:21 -07:00
local nomsu
2018-07-19 20:41:31 -07:00
if key.type == " Text " and # key == 1 and Parser.is_identifier ( key [ 1 ] ) then
2018-07-20 17:51:21 -07:00
nomsu = NomsuCode ( key.source , key [ 1 ] )
2018-07-19 20:41:31 -07:00
else
2018-07-20 17:51:21 -07:00
nomsu = recurse ( key )
2018-07-19 20:41:31 -07:00
end
if key.type == " Action " or key.type == " Block " then
2018-07-20 17:51:21 -07:00
nomsu : parenthesize ( )
2018-07-19 20:41:31 -07:00
end
2018-07-20 17:51:21 -07:00
assert ( value.type ~= " Block " , " Didn't expect to find a Block as a value in a dict " )
nomsu : append ( " : " )
2018-07-19 20:41:31 -07:00
if value then
2018-07-20 17:51:21 -07:00
local value_nomsu = recurse ( value , nomsu )
if value.type == " Block " then
value_nomsu : parenthesize ( )
end
nomsu : append ( value_nomsu )
2018-07-19 20:41:31 -07:00
end
2018-07-20 17:51:21 -07:00
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-19 20:41:31 -07:00
end
2018-07-20 17:51:21 -07:00
return nomsu
2018-07-19 20:41:31 -07:00
elseif " IndexChain " == _exp_0 then
local nomsu = NomsuCode ( tree.source )
for i , bit in ipairs ( tree ) do
if i > 1 then
nomsu : append ( " . " )
end
local bit_nomsu
2018-07-26 15:57:11 -07:00
if i > 1 and bit.type == " Text " and # bit == 1 and type ( bit [ 1 ] ) == ' string ' and Parser.is_identifier ( bit [ 1 ] ) then
2018-07-19 20:41:31 -07:00
bit_nomsu = bit [ 1 ]
else
2018-07-20 17:51:21 -07:00
bit_nomsu = recurse ( bit , nomsu )
2018-07-19 20:41:31 -07:00
end
2018-07-20 17:51:21 -07:00
assert ( bit.type ~= " Block " )
if bit.type == " Action " or bit.type == " IndexChain " or ( bit.type == " Number " and i < # tree ) then
2018-07-19 20:41:31 -07:00
bit_nomsu : parenthesize ( )
end
nomsu : append ( bit_nomsu )
2018-07-20 17:51:21 -07:00
if check then
2018-07-20 19:36:05 -07:00
check ( len , nomsu , tree )
2018-07-20 17:51:21 -07:00
end
2018-07-19 20:41:31 -07:00
end
return nomsu
elseif " Number " == _exp_0 then
return NomsuCode ( tree.source , tostring ( tree [ 1 ] ) )
elseif " Var " == _exp_0 then
return NomsuCode ( tree.source , " % " , tree [ 1 ] )
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
end
end
2018-07-20 17:51:21 -07:00
NomsuCompiler.tree_to_nomsu = function ( self , tree , pop_comments )
if pop_comments == nil then
pop_comments = nil
2018-07-19 21:09:44 -07:00
end
2018-07-20 17:51:21 -07:00
if not ( pop_comments ) then
local comment_set = { }
2018-07-17 14:12:11 -07:00
local find_comments
find_comments = function ( t )
if t.comments and t.source . filename == tree.source . filename then
local _list_0 = t.comments
for _index_0 = 1 , # _list_0 do
local c = _list_0 [ _index_0 ]
2018-07-20 17:51:21 -07:00
comment_set [ c ] = true
2018-07-14 14:52:28 -07:00
end
2018-07-12 21:42:09 -07:00
end
2018-07-19 22:59:29 -07:00
local _list_0 = t
for _index_0 = 1 , # _list_0 do
local x = _list_0 [ _index_0 ]
2018-07-17 14:12:11 -07:00
if AST.is_syntax_tree ( x ) then
find_comments ( x )
2018-07-14 14:41:17 -07:00
end
end
2018-07-17 14:12:11 -07:00
end
2018-07-19 21:28:01 -07:00
find_comments ( tree )
2018-07-20 17:51:21 -07:00
local comments
do
local _accum_0 = { }
local _len_0 = 1
for c in pairs ( comment_set ) do
_accum_0 [ _len_0 ] = c
_len_0 = _len_0 + 1
end
comments = _accum_0
end
2018-07-19 21:28:01 -07:00
table.sort ( comments , function ( a , b )
return ( a.pos > b.pos )
end )
2018-07-20 17:51:21 -07:00
pop_comments = function ( pos , prefix , suffix )
if prefix == nil then
prefix = ' '
2018-07-15 19:41:22 -07:00
end
2018-07-20 17:51:21 -07:00
if suffix == nil then
suffix = ' '
2018-07-17 14:12:11 -07:00
end
2018-07-20 17:51:21 -07:00
local nomsu = NomsuCode ( tree.source )
for i = # comments , 1 , - 1 do
if comments [ i ] . pos > pos then
break
end
local comment
comment , comments [ i ] = comments [ i ] . comment , nil
nomsu : append ( " # " .. ( gsub ( comment , " \n " , " \n " ) ) .. " \n " )
if comment : match ( " ^ \n . " ) then
nomsu : append ( " \n " )
end
end
if # nomsu.bits == 0 then
return ' '
end
if not ( prefix == ' ' ) then
nomsu : prepend ( prefix )
end
if not ( suffix == ' ' ) then
nomsu : append ( suffix )
end
return nomsu
2018-07-17 14:12:11 -07:00
end
2018-07-20 17:51:21 -07:00
end
local recurse
recurse = function ( t , pos )
if pos == nil then
pos = 0
end
if type ( pos ) ~= ' number ' then
pos = # tostring ( pos ) : match ( " [ ]*([^ \n ]*)$ " )
end
local space = MAX_LINE - pos
local inline
2018-07-20 19:36:05 -07:00
for prefix , nomsu , tree in coroutine.wrap ( function ( )
2018-07-20 17:51:21 -07:00
inline = self : tree_to_inline_nomsu ( t , false , coroutine.yield )
end ) do
2018-07-20 19:36:05 -07:00
local len = # tostring ( nomsu )
if prefix + len > MAX_LINE then
break
end
if tree.type == " Block " and ( # tree > 1 or len > 20 ) then
2018-07-20 17:51:21 -07:00
break
end
2018-07-20 19:36:05 -07:00
if tree.type == " Text " then
2018-07-20 19:47:58 -07:00
local check_for_nl
check_for_nl = function ( tree )
2018-07-20 19:36:05 -07:00
local found_nl = false
for i , b in ipairs ( tree ) do
2018-07-20 19:47:58 -07:00
if type ( b ) ~= ' string ' and b.type == " Text " and check_for_nl ( b ) then
return true
end
2018-07-20 19:36:05 -07:00
if i == 1 and type ( b ) == ' string ' then
b = b : match ( ' ^[ \n ]*(.*) ' )
end
found_nl = found_nl or ( type ( b ) == ' string ' and b : match ( ' \n ' ) )
if found_nl and ( type ( b ) ~= ' string ' or b : match ( ' [^ \n ] ' ) ) then
return true
end
end
2018-07-20 19:47:58 -07:00
end
if check_for_nl ( tree ) then
2018-07-20 19:36:05 -07:00
break
end
end
2018-06-28 14:12:24 -07:00
end
2018-07-20 17:51:21 -07:00
if inline and # tostring ( inline ) <= space then
return inline
2018-07-19 22:59:29 -07:00
end
2018-07-20 17:51:21 -07:00
local indented = self : tree_to_nomsu ( t , pop_comments , space )
if t.type == " Action " and not ( tree.type == " Block " or tree.type == " FileChunks " ) then
2018-07-20 18:10:16 -07:00
indented = NomsuCode ( t.source , " (..) \n " , pop_comments ( t.source . start ) , indented )
2018-07-19 22:59:29 -07:00
end
2018-07-20 17:51:21 -07:00
return indented
2018-07-14 14:41:17 -07:00
end
2018-06-19 01:27:32 -07:00
local _exp_0 = tree.type
2018-07-15 19:41:22 -07:00
if " FileChunks " == _exp_0 then
2018-07-17 14:12:11 -07:00
local nomsu = NomsuCode ( tree.source , pop_comments ( tree.source . start ) )
2018-07-30 15:05:41 -07:00
local should_clump
should_clump = function ( prev_line , line )
2018-07-30 15:12:11 -07:00
if prev_line.type == " Action " and line.type == " Action " then
2018-08-28 15:08:00 -07:00
if prev_line.stub == " use 1 " then
return line.stub == " use 1 "
2018-07-30 15:05:41 -07:00
end
2018-08-28 15:08:00 -07:00
if prev_line.stub == " test 1 " then
2018-07-30 15:05:41 -07:00
return true
end
2018-08-28 15:08:00 -07:00
if line.stub == " test 1 " then
2018-07-30 15:05:41 -07:00
return false
end
end
2018-07-30 15:12:11 -07:00
return not recurse ( prev_line ) : is_multiline ( )
2018-07-30 15:05:41 -07:00
end
2018-07-19 21:28:01 -07:00
for chunk_no , chunk in ipairs ( tree ) do
if chunk_no > 1 then
2018-06-28 14:12:24 -07:00
nomsu : append ( " \n \n " .. tostring ( ( " ~ " ) : rep ( 80 ) ) .. " \n \n " )
2018-06-19 01:27:32 -07:00
end
2018-06-28 14:12:24 -07:00
nomsu : append ( pop_comments ( chunk.source . start ) )
2018-07-19 21:09:44 -07:00
if chunk.type == " Block " then
2018-07-19 21:28:01 -07:00
for line_no , line in ipairs ( chunk ) do
2018-07-30 15:05:41 -07:00
if line_no > 1 then
if should_clump ( chunk [ line_no - 1 ] , line ) then
nomsu : append ( " \n " , pop_comments ( line.source . start , ' \n ' ) )
else
nomsu : append ( " \n \n " , pop_comments ( line.source . start ) )
end
2018-07-19 21:09:44 -07:00
end
2018-07-30 15:05:41 -07:00
nomsu : append ( self : tree_to_nomsu ( line , pop_comments ) )
2018-07-19 21:09:44 -07:00
end
2018-07-19 21:28:01 -07:00
nomsu : append ( pop_comments ( chunk.source . stop , ' \n ' ) )
2018-07-19 21:09:44 -07:00
else
nomsu : append ( recurse ( chunk ) )
end
2018-06-28 14:12:24 -07:00
end
2018-07-17 14:12:11 -07:00
nomsu : append ( pop_comments ( tree.source . stop , ' \n ' ) )
2018-08-28 15:08:00 -07:00
if not ( nomsu : match ( " \n $ " ) ) then
2018-07-22 14:57:37 -07:00
nomsu : append ( ' \n ' )
end
2018-06-19 01:27:32 -07:00
return nomsu
elseif " Action " == _exp_0 then
2018-07-19 22:59:29 -07:00
local pos , next_space = tree.source . start , ' '
2018-07-19 20:41:31 -07:00
local nomsu = NomsuCode ( tree.source , pop_comments ( pos ) )
2018-08-28 15:08:00 -07:00
if tree.target then
nomsu : append ( self : tree_to_nomsu ( tree.target ) , " :: " )
end
2018-07-19 20:41:31 -07:00
for i , bit in ipairs ( tree ) do
2018-07-19 22:59:29 -07:00
if next_space == " \n .. " or ( next_space == " " and nomsu : trailing_line_len ( ) > MAX_LINE ) then
nomsu : append ( " \n " , pop_comments ( pos ) , ' .. ' )
next_space = " "
2018-07-19 20:41:31 -07:00
end
if type ( bit ) == " string " then
2018-07-19 22:59:29 -07:00
if not ( type ( tree [ i - 1 ] ) == ' string ' and Parser.is_operator ( tree [ i - 1 ] ) ~= Parser.is_operator ( bit ) ) then
nomsu : append ( next_space )
2018-07-19 20:41:31 -07:00
end
2018-07-19 22:59:29 -07:00
nomsu : append ( bit )
next_space = ' '
2018-07-20 17:51:21 -07:00
elseif bit.type == " Block " then
2018-08-28 15:08:00 -07:00
nomsu : append ( recurse ( bit , # nomsu : match ( ' [^ \n ]*$ ' ) ) )
2018-07-20 17:51:21 -07:00
pos = bit.source . stop
next_space = inline and " " or " \n .. "
2018-07-19 20:41:31 -07:00
else
2018-07-20 17:51:21 -07:00
nomsu : append ( next_space )
2018-08-28 15:08:00 -07:00
local bit_nomsu = recurse ( bit , # nomsu : match ( ' [^ \n ]*$ ' ) )
2018-07-20 17:51:21 -07:00
if bit.type == " Action " and not bit_nomsu : is_multiline ( ) then
bit_nomsu : parenthesize ( )
2018-07-15 19:41:22 -07:00
end
2018-07-20 17:51:21 -07:00
nomsu : append ( bit_nomsu )
2018-07-19 20:41:31 -07:00
pos = bit.source . stop
2018-07-20 17:51:21 -07:00
next_space = bit_nomsu : is_multiline ( ) and " \n .. " or " "
2018-06-19 01:27:32 -07:00
end
end
2018-07-19 22:59:29 -07:00
nomsu : append ( pop_comments ( tree.source . stop , ' \n ' ) )
2018-07-19 20:41:31 -07:00
return nomsu
2018-06-19 01:27:32 -07:00
elseif " EscapedNomsu " == _exp_0 then
2018-07-20 17:51:21 -07:00
return NomsuCode ( tree.source , " \\ " , recurse ( tree [ 1 ] , 1 ) )
2018-06-19 01:27:32 -07:00
elseif " Block " == _exp_0 then
2018-07-17 14:12:11 -07:00
local nomsu = NomsuCode ( tree.source , pop_comments ( tree.source . start ) )
2018-06-19 01:27:32 -07:00
for i , line in ipairs ( tree ) do
2018-07-23 14:40:20 -07:00
nomsu : append ( pop_comments ( line.source . start , i > 1 and ' \n ' or ' ' ) )
2018-07-19 20:41:31 -07:00
local line_nomsu = recurse ( line )
nomsu : append ( line_nomsu )
2018-06-19 01:27:32 -07:00
if i < # tree then
2018-08-28 15:08:00 -07:00
nomsu : append ( line_nomsu : match ( ' \n [^ \n ]* \n ' ) and " \n \n " or " \n " )
2018-06-19 01:27:32 -07:00
end
end
2018-07-17 14:12:11 -07:00
nomsu : append ( pop_comments ( tree.source . stop , ' \n ' ) )
2018-07-19 21:09:44 -07:00
return NomsuCode ( tree.source , " : \n " , nomsu )
2018-06-19 01:27:32 -07:00
elseif " Text " == _exp_0 then
2018-07-20 20:13:01 -07:00
local max_line = math.floor ( 1.5 * MAX_LINE )
2018-07-20 17:51:21 -07:00
local add_text
add_text = function ( nomsu , tree )
2018-07-19 20:41:31 -07:00
for i , bit in ipairs ( tree ) do
if type ( bit ) == ' string ' then
bit = Parser.escape ( bit )
2018-07-21 14:43:49 -07:00
local bit_lines = Files.get_lines ( bit )
2018-07-19 20:41:31 -07:00
for j , line in ipairs ( bit_lines ) do
if j > 1 then
nomsu : append ( " \n " )
2018-07-20 20:13:01 -07:00
elseif # line > 10 and nomsu : trailing_line_len ( ) > max_line then
2018-07-20 17:51:21 -07:00
nomsu : append ( " \\ \n .. " )
2018-06-19 01:27:32 -07:00
end
2018-07-20 17:51:21 -07:00
while # line > 0 do
2018-07-20 20:13:01 -07:00
local space = max_line - nomsu : trailing_line_len ( )
local split = find ( line , " [%p%s] " , space )
2018-07-20 17:51:21 -07:00
if not split or split > space + 10 then
split = space + 10
end
if # line - split < 10 then
split = # line
end
local bite
bite , line = sub ( line , 1 , split ) , sub ( line , split + 1 , - 1 )
nomsu : append ( bite )
if # line > 0 then
nomsu : append ( " \\ \n .. " )
2018-06-19 01:27:32 -07:00
end
end
2018-07-19 20:41:31 -07:00
end
elseif bit.type == " Text " then
2018-07-20 17:51:21 -07:00
add_text ( nomsu , bit )
2018-07-19 20:41:31 -07:00
else
2018-07-20 17:51:21 -07:00
nomsu : append ( " \\ " )
2018-08-28 15:08:00 -07:00
local interp_nomsu = recurse ( bit , # nomsu : match ( ' [^ \n ]*$ ' ) )
2018-07-20 17:51:21 -07:00
if not ( interp_nomsu : is_multiline ( ) ) then
2018-07-19 22:59:29 -07:00
if bit.type == " Var " then
2018-07-19 20:41:31 -07:00
if type ( tree [ i + 1 ] ) == ' string ' and not match ( tree [ i + 1 ] , " ^[ \n \t ,.:;#(){}[%]] " ) then
2018-07-17 23:33:49 -07:00
interp_nomsu : parenthesize ( )
2018-07-11 14:13:43 -07:00
end
2018-07-20 17:51:21 -07:00
elseif bit.type ~= " List " and bit.type ~= " Dict " then
interp_nomsu : parenthesize ( )
2018-06-19 01:27:32 -07:00
end
end
2018-07-20 17:51:21 -07:00
nomsu : append ( interp_nomsu )
2018-07-30 14:26:08 -07:00
if interp_nomsu : is_multiline ( ) and i < # tree then
2018-07-20 17:51:21 -07:00
nomsu : append ( " \n .. " )
end
2018-06-19 01:27:32 -07:00
end
end
end
2018-07-20 17:51:21 -07:00
local nomsu = NomsuCode ( tree.source )
add_text ( nomsu , tree )
2018-08-28 15:08:00 -07:00
if nomsu : is_multiline ( ) and nomsu : match ( " \n $ " ) then
2018-07-20 19:36:05 -07:00
nomsu : append ( ' \\ ("") ' )
end
2018-07-20 17:51:21 -07:00
return NomsuCode ( tree.source , ' ".." \n ' , nomsu )
2018-07-22 15:59:29 -07:00
elseif " List " == _exp_0 or " Dict " == _exp_0 then
2018-07-19 20:41:31 -07:00
assert ( # tree > 0 )
local nomsu = NomsuCode ( tree.source , pop_comments ( tree [ 1 ] . source.start ) )
for i , item in ipairs ( tree ) do
2018-07-20 17:51:21 -07:00
if nomsu : trailing_line_len ( ) == 0 then
nomsu : append ( pop_comments ( item.source . start ) )
2018-06-19 01:27:32 -07:00
end
2018-07-20 17:51:21 -07:00
local inline_nomsu = self : tree_to_inline_nomsu ( item )
2018-08-28 15:08:00 -07:00
local item_nomsu = # tostring ( inline_nomsu ) <= MAX_LINE and inline_nomsu or recurse ( item , # nomsu : match ( ' [^ \n ]*$ ' ) )
2018-07-20 17:51:21 -07:00
nomsu : append ( item_nomsu )
if i < # tree then
nomsu : append ( ( item_nomsu : is_multiline ( ) or nomsu : trailing_line_len ( ) + # tostring ( item_nomsu ) >= MAX_LINE ) and ' \n ' or ' , ' )
2018-06-19 01:27:32 -07:00
end
end
2018-07-19 20:41:31 -07:00
nomsu : append ( pop_comments ( tree.source . stop , ' \n ' ) )
2018-07-22 15:59:29 -07:00
if tree.type == " List " then
return NomsuCode ( tree.source , " [..] \n " , nomsu )
else
return NomsuCode ( tree.source , " {..} \n " , nomsu )
2018-06-19 01:27:32 -07:00
end
elseif " DictEntry " == _exp_0 then
local key , value = tree [ 1 ] , tree [ 2 ]
2018-07-20 17:51:21 -07:00
local nomsu
2018-07-17 23:33:49 -07:00
if key.type == " Text " and # key == 1 and Parser.is_identifier ( key [ 1 ] ) then
2018-07-20 17:51:21 -07:00
nomsu = NomsuCode ( key.source , key [ 1 ] )
2018-07-17 23:33:49 -07:00
else
2018-07-20 17:51:21 -07:00
nomsu = self : tree_to_inline_nomsu ( key )
2018-07-17 23:33:49 -07:00
end
2018-06-19 01:27:32 -07:00
if key.type == " Action " or key.type == " Block " then
2018-07-20 17:51:21 -07:00
nomsu : parenthesize ( )
2018-06-19 01:27:32 -07:00
end
2018-07-20 17:51:21 -07:00
nomsu : append ( " : " , recurse ( value , # tostring ( nomsu ) ) )
return nomsu
2018-07-19 20:41:31 -07:00
elseif " IndexChain " == _exp_0 or " Number " == _exp_0 or " Var " == _exp_0 then
return self : tree_to_inline_nomsu ( tree )
2018-06-19 01:27:32 -07:00
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
end
end
end
return NomsuCompiler