2018-05-15 16:36:21 -07:00
local _pairs , _ipairs = pairs , ipairs
2018-05-09 13:34:33 -07:00
if jit then
2018-05-09 20:34:32 -07:00
package.cpath = " ./luajit_lpeg/?.so; " .. package.cpath
2018-05-09 13:34:33 -07:00
bit32 = require ( ' bit ' )
pairs = function ( x )
do
local mt = getmetatable ( x )
if mt then
if mt.__pairs then
return mt.__pairs ( x )
end
end
end
return _pairs ( x )
end
ipairs = function ( x )
do
local mt = getmetatable ( x )
if mt then
if mt.__ipairs then
return mt.__ipairs ( x )
end
end
end
return _ipairs ( x )
end
end
2018-05-09 20:34:32 -07:00
re = require ( ' re ' )
lpeg = require ( ' lpeg ' )
2018-04-11 20:05:12 -07:00
lpeg.setmaxstack ( 10000 )
2018-05-24 14:57:24 -07:00
local P , R , V , S , Cg , C , Cp , B
P , R , V , S , Cg , C , Cp , B = lpeg.P , lpeg.R , lpeg.V , lpeg.S , lpeg.Cg , lpeg.C , lpeg.Cp , lpeg.B
2017-12-18 16:26:26 -08:00
local utils = require ( ' utils ' )
2018-01-11 03:32:12 -08:00
local new_uuid = require ( ' uuid ' )
2018-02-13 15:17:45 -08:00
local immutable = require ( ' immutable ' )
2018-04-17 14:18:23 -07:00
Tuple = immutable ( nil , {
name = " Tuple "
} )
2017-12-18 16:25:56 -08:00
local repr , stringify , min , max , equivalent , set , is_list , sum
repr , stringify , min , max , equivalent , set , is_list , sum = utils.repr , utils.stringify , utils.min , utils.max , utils.equivalent , utils.set , utils.is_list , utils.sum
2017-10-08 20:41:05 -07:00
local colors = setmetatable ( { } , {
__index = function ( )
return " "
end
} )
2018-04-19 17:23:44 -07:00
colored = setmetatable ( { } , {
2017-09-28 17:49:15 -07:00
__index = function ( _ , color )
return ( function ( msg )
2018-04-11 20:05:12 -07:00
return colors [ color ] .. tostring ( msg or ' ' ) .. colors.reset
2017-09-28 17:49:15 -07:00
end )
end
} )
2017-09-21 21:11:13 -07:00
local insert , remove , concat
do
local _obj_0 = table
insert , remove , concat = _obj_0.insert , _obj_0.remove , _obj_0.concat
end
2018-04-08 18:11:44 -07:00
local debug_getinfo = debug.getinfo
2018-04-18 15:28:46 -07:00
local Nomsu , Lua , Source
2018-04-11 21:07:13 -07:00
do
2018-04-26 14:00:01 -07:00
local _obj_0 = require ( " code_obj " )
2018-04-18 15:28:46 -07:00
Nomsu , Lua , Source = _obj_0.Nomsu , _obj_0.Lua , _obj_0.Source
2018-04-11 21:07:13 -07:00
end
2018-04-28 18:07:14 -07:00
local STDIN , STDOUT , STDERR = " /dev/fd/0 " , " /dev/fd/1 " , " /dev/fd/2 "
2018-04-13 14:54:35 -07:00
FILE_CACHE = setmetatable ( { } , {
2018-04-11 20:05:12 -07:00
__index = function ( self , filename )
local file = io.open ( filename )
if not ( file ) then
return nil
end
2018-05-09 13:34:33 -07:00
local contents = file : read ( " *a " )
2018-04-11 20:05:12 -07:00
file : close ( )
2018-04-20 16:23:53 -07:00
self [ filename ] = contents
return contents
2018-04-11 20:05:12 -07:00
end
} )
2018-04-28 17:08:28 -07:00
local iterate_single
iterate_single = function ( item , prev )
if item == prev then
return nil
else
return item
end
end
local all_files
all_files = function ( path )
2018-04-28 18:07:14 -07:00
if path : match ( " %.nom$ " ) or path : match ( " %.lua$ " ) or path : match ( " ^/dev/fd/[012]$ " ) then
2018-04-28 17:08:28 -07:00
return iterate_single , path
end
path = path : gsub ( " \\ " , " \\ \\ " ) : gsub ( " ` " , " " ) : gsub ( ' " ' , ' \\ " ' ) : gsub ( " $ " , " " )
2018-05-24 15:51:06 -07:00
return coroutine.wrap ( function ( )
2018-05-24 16:13:23 -07:00
local f = io.popen ( ' find -L " ' .. path .. ' " -not -path "*/ \\ .*" -type f -name "*.nom" ' )
2018-05-24 15:51:06 -07:00
for line in f : lines ( ) do
coroutine.yield ( line )
end
local success = f : close ( )
if not ( success ) then
return error ( " Invalid file path: " .. tostring ( path ) )
end
end )
2018-04-28 17:08:28 -07:00
end
2018-04-11 20:05:12 -07:00
local line_counter = re.compile ( [ [ lines <- { | line ( % nl line ) * | }
line <- { } ( ! % nl . ) *
] ] , {
nl = P ( " \r " ) ^ - 1 * P ( " \n " )
} )
2018-04-11 21:07:13 -07:00
LINE_STARTS = setmetatable ( { } , {
2018-04-11 20:05:12 -07:00
__mode = " k " ,
__index = function ( self , k )
2018-04-18 15:45:58 -07:00
if type ( k ) ~= ' string ' then
k = tostring ( k )
do
local v = rawget ( self , k )
if v then
return v
end
end
end
local line_starts = line_counter : match ( k )
2018-04-11 20:05:12 -07:00
self [ k ] = line_starts
return line_starts
end
} )
local LUA_METADATA = { }
local lua_line_to_nomsu_line
lua_line_to_nomsu_line = function ( lua_filename , lua_line_no )
local metadata = assert ( LUA_METADATA [ lua_filename ] , " Failed to find nomsu metadata for: " .. tostring ( lua_filename ) .. " . " )
local lua_offset = LINE_STARTS [ metadata.lua_file ] [ lua_line_no ]
local best = metadata.nomsu_sources [ 1 ]
for lua , nomsu in pairs ( metadata.lua_to_nomsu ) do
if lua.start <= lua_offset and lua > best then
best = lua
end
end
return best : get_line_number ( )
end
2018-01-23 19:22:20 -08:00
do
local STRING_METATABLE = getmetatable ( " " )
STRING_METATABLE.__add = function ( self , other )
return self .. stringify ( other )
end
2018-01-24 13:13:03 -08:00
STRING_METATABLE.__index = function ( self , i )
2018-05-14 14:45:38 -07:00
local ret = string [ i ]
if ret ~= nil then
return ret
end
2018-01-24 13:13:03 -08:00
if type ( i ) == ' number ' then
return string.sub ( self , i , i )
elseif type ( i ) == ' table ' then
return string.sub ( self , i [ 1 ] , i [ 2 ] )
end
end
2018-01-23 19:22:20 -08:00
end
2018-04-17 14:18:23 -07:00
local Types = require ( " nomsu_tree " )
2018-01-19 17:28:40 -08:00
local NOMSU_DEFS
2017-12-30 14:31:07 -08:00
do
2018-01-19 17:28:40 -08:00
local _with_0 = { }
2018-01-25 17:34:49 -08:00
_with_0.nl = P ( " \r " ) ^ - 1 * P ( " \n " )
2018-01-19 17:28:40 -08:00
_with_0.ws = S ( " \t " )
_with_0.tonumber = tonumber
_with_0.print = function ( src , pos , msg )
print ( msg , pos , repr ( src : sub ( math.max ( 0 , pos - 16 ) , math.max ( 0 , pos - 1 ) ) .. " | " .. src : sub ( pos , pos + 16 ) ) )
return true
end
local string_escapes = {
n = " \n " ,
t = " \t " ,
b = " \b " ,
a = " \a " ,
v = " \v " ,
f = " \f " ,
r = " \r "
}
local digit , hex = R ( ' 09 ' ) , R ( ' 09 ' , ' af ' , ' AF ' )
_with_0.escaped_char = ( P ( " \\ " ) * S ( " xX " ) * C ( hex * hex ) ) / function ( self )
return string.char ( tonumber ( self , 16 ) )
end
_with_0.escaped_char = _with_0.escaped_char + ( ( P ( " \\ " ) * C ( digit * ( digit ^ - 2 ) ) ) / function ( self )
return string.char ( tonumber ( self ) )
end )
_with_0.escaped_char = _with_0.escaped_char + ( ( P ( " \\ " ) * C ( S ( " ntbavfr " ) ) ) / string_escapes )
_with_0.operator_char = S ( " '~`!@$^&*-+=|<>?/ " )
_with_0.operator = _with_0.operator_char ^ 1
_with_0.utf8_char = ( R ( " \194 \223 " ) * R ( " \128 \191 " ) + R ( " \224 \239 " ) * R ( " \128 \191 " ) * R ( " \128 \191 " ) + R ( " \240 \244 " ) * R ( " \128 \191 " ) * R ( " \128 \191 " ) * R ( " \128 \191 " ) )
_with_0.ident_char = R ( " az " , " AZ " , " 09 " ) + P ( " _ " ) + _with_0.utf8_char
_with_0.indent = P ( function ( self , start )
2018-01-30 15:10:21 -08:00
local nodent = lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ]
local indented = nodent .. " "
if self : sub ( start , start + # indented - 1 ) == indented then
insert ( lpeg.userdata . indent_stack , indented )
return start + # indented
2017-10-13 16:10:47 -07:00
end
2017-12-30 14:31:07 -08:00
end )
2018-01-19 17:28:40 -08:00
_with_0.dedent = P ( function ( self , start )
2018-01-30 15:10:21 -08:00
local nodent = lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ]
2018-05-26 13:09:20 -07:00
local spaces = self : match ( " ^[ ]* " , start )
2018-01-30 15:10:21 -08:00
if # spaces <= # nodent - 4 then
2018-01-19 17:28:40 -08:00
remove ( lpeg.userdata . indent_stack )
2017-12-30 14:31:07 -08:00
return start
2017-09-22 00:03:32 -07:00
end
2017-12-30 14:31:07 -08:00
end )
2018-01-19 17:28:40 -08:00
_with_0.nodent = P ( function ( self , start )
2018-01-30 15:10:21 -08:00
local nodent = lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ]
if self : sub ( start , start + # nodent - 1 ) == nodent then
return start + # nodent
2017-09-22 00:03:32 -07:00
end
2017-12-30 14:31:07 -08:00
end )
2018-05-03 16:30:55 -07:00
_with_0.error = function ( src , end_pos , start_pos , err_msg )
local seen_errors = lpeg.userdata . errors
if seen_errors [ start_pos ] then
return true
2018-01-19 17:28:40 -08:00
end
2018-05-03 16:30:55 -07:00
local err_pos = start_pos
local text_loc = lpeg.userdata . source_code.source : sub ( err_pos , err_pos )
local line_no = text_loc : get_line_number ( )
2018-04-19 17:23:44 -07:00
src = FILE_CACHE [ text_loc.filename ]
2018-05-03 16:30:55 -07:00
local prev_line = line_no == 1 and " " or src : sub ( LINE_STARTS [ src ] [ line_no - 1 ] or 1 , LINE_STARTS [ src ] [ line_no ] - 2 )
2018-04-27 16:45:11 -07:00
local err_line = src : sub ( LINE_STARTS [ src ] [ line_no ] , ( LINE_STARTS [ src ] [ line_no + 1 ] or 0 ) - 2 )
local next_line = src : sub ( LINE_STARTS [ src ] [ line_no + 1 ] or - 1 , ( LINE_STARTS [ src ] [ line_no + 2 ] or 0 ) - 2 )
2018-05-03 16:30:55 -07:00
local pointer = ( " - " ) : rep ( err_pos - LINE_STARTS [ src ] [ line_no ] ) .. " ^ "
err_msg = ( err_msg or " Parse error " ) .. " at " .. tostring ( lpeg.userdata . source_code.source . filename ) .. " : " .. tostring ( line_no ) .. " : \n "
if # prev_line > 0 then
err_msg = err_msg .. ( " \n " .. prev_line )
end
err_msg = err_msg .. " \n " .. tostring ( err_line ) .. " \n " .. tostring ( pointer )
if # next_line > 0 then
err_msg = err_msg .. ( " \n " .. next_line )
end
seen_errors [ start_pos ] = err_msg
return true
2018-01-19 17:28:40 -08:00
end
NOMSU_DEFS = _with_0
end
setmetatable ( NOMSU_DEFS , {
__index = function ( self , key )
local make_node
2018-05-26 15:04:31 -07:00
make_node = function ( start , ... )
local args = {
...
}
local stop = args [ # args ]
local source = lpeg.userdata . source : sub ( start , stop )
args [ # args ] = nil
local tree
if Types [ key ] . is_multi then
tree = Types [ key ] ( Tuple ( unpack ( args ) ) , source )
else
tree = Types [ key ] ( args [ 1 ] , source )
end
2018-05-24 20:27:08 -07:00
return tree
2017-09-22 00:03:32 -07:00
end
2018-01-19 17:28:40 -08:00
self [ key ] = make_node
return make_node
end
} )
2018-04-11 20:05:12 -07:00
local NOMSU_PATTERN
2018-01-19 17:28:40 -08:00
do
2017-12-30 14:31:07 -08:00
local peg_tidier = re.compile ( [ [ file <- { ~ % nl * ( def / comment ) ( % nl + ( def / comment ) ) * % nl * ~ }
def <- anon_def / captured_def
anon_def <- ( { ident } ( " " * ) " : "
{ ( ( % nl " " + [ ^% nl ] * ) + ) / ( [ ^% nl ] * ) } ) -> " %1 <- %2 "
captured_def <- ( { ident } ( " " * ) " ( " { ident } " ) " ( " " * ) " : "
2018-05-26 15:04:31 -07:00
{ ( ( % nl " " + [ ^% nl ] * ) + ) / ( [ ^% nl ] * ) } ) -> " %1 <- (({} %3 {}) -> %2) "
2017-12-30 14:31:07 -08:00
ident <- [ a - zA - Z_ ] [ a - zA - Z0 - 9 _ ] *
comment <- " -- " [ ^% nl ] *
] ] )
2018-04-11 20:05:12 -07:00
local nomsu_peg = peg_tidier : match ( FILE_CACHE [ " nomsu.peg " ] )
NOMSU_PATTERN = re.compile ( nomsu_peg , NOMSU_DEFS )
2017-12-30 14:31:07 -08:00
end
2017-09-13 16:22:04 -07:00
local NomsuCompiler
2017-09-12 21:38:54 -07:00
do
local _class_0
2018-05-16 19:08:16 -07:00
local stub_defs , stub_pattern , var_pattern , _nomsu_chunk_counter , _running_files , MAX_LINE , math_expression
2017-09-12 21:38:54 -07:00
local _base_0 = {
2018-05-03 22:33:44 -07:00
define_action = function ( self , signature , fn , is_compile_action )
if is_compile_action == nil then
is_compile_action = false
2017-09-12 21:38:54 -07:00
end
2018-05-03 21:55:46 -07:00
assert ( type ( fn ) == ' function ' , " Bad fn: " .. tostring ( repr ( fn ) ) )
2017-10-13 16:10:47 -07:00
if type ( signature ) == ' string ' then
2018-01-25 17:34:49 -08:00
signature = {
2017-10-13 16:10:47 -07:00
signature
2018-01-25 17:34:49 -08:00
}
2018-05-03 21:55:46 -07:00
elseif type ( signature ) ~= ' table ' then
2018-01-25 17:34:49 -08:00
error ( " Invalid signature, expected list of strings, but got: " .. tostring ( repr ( signature ) ) , 0 )
2017-10-13 16:10:47 -07:00
end
2018-05-03 21:55:46 -07:00
local fn_info = debug_getinfo ( fn , " u " )
assert ( not fn_info.isvararg , " Vararg functions aren't supported. Sorry, use a list instead. " )
local fn_arg_positions
do
local _tbl_0 = { }
for i = 1 , fn_info.nparams do
_tbl_0 [ debug.getlocal ( fn , i ) ] = i
end
fn_arg_positions = _tbl_0
end
local arg_orders = { }
for _index_0 = 1 , # signature do
local alias = signature [ _index_0 ]
local stub = assert ( stub_pattern : match ( alias ) )
2018-05-15 15:21:32 -07:00
local stub_args = assert ( var_pattern : match ( alias ) ) ;
2018-05-03 22:33:44 -07:00
( is_compile_action and self.environment . COMPILE_ACTIONS or self.environment . ACTIONS ) [ stub ] = fn
2018-05-03 21:55:46 -07:00
do
local _accum_0 = { }
local _len_0 = 1
for _index_1 = 1 , # stub_args do
local a = stub_args [ _index_1 ]
2018-05-26 15:04:31 -07:00
_accum_0 [ _len_0 ] = fn_arg_positions [ Types.Var . as_lua_id ( {
value = a
} ) ]
2018-05-03 21:55:46 -07:00
_len_0 = _len_0 + 1
2017-12-04 17:35:47 -08:00
end
2018-05-03 21:55:46 -07:00
arg_orders [ stub ] = _accum_0
2017-12-04 17:35:47 -08:00
end
end
2018-05-03 21:55:46 -07:00
self.environment . ARG_ORDERS [ fn ] = arg_orders
2017-12-04 17:35:47 -08:00
end ,
2018-05-03 21:55:46 -07:00
define_compile_action = function ( self , signature , fn )
return self : define_action ( signature , fn , true )
2017-12-04 17:35:47 -08:00
end ,
2018-04-18 15:28:46 -07:00
parse = function ( self , nomsu_code )
2018-04-19 19:43:23 -07:00
if type ( nomsu_code ) == ' string ' then
_nomsu_chunk_counter = _nomsu_chunk_counter + 1
local filename = " <nomsu chunk # " .. tostring ( _nomsu_chunk_counter ) .. " >.nom "
FILE_CACHE [ filename ] = nomsu_code
2018-04-20 16:23:53 -07:00
nomsu_code = Nomsu ( filename , nomsu_code )
2018-04-19 19:43:23 -07:00
end
2018-02-08 16:22:57 -08:00
local userdata = {
source_code = nomsu_code ,
indent_stack = {
" "
2018-05-03 16:30:55 -07:00
} ,
2018-05-24 20:27:08 -07:00
errors = { } ,
2018-05-26 15:04:31 -07:00
source = nomsu_code.source
2018-02-08 16:22:57 -08:00
}
2018-01-19 17:28:40 -08:00
local old_userdata
old_userdata , lpeg.userdata = lpeg.userdata , userdata
2018-04-18 15:28:46 -07:00
local tree = NOMSU_PATTERN : match ( tostring ( nomsu_code ) )
2018-01-19 17:28:40 -08:00
lpeg.userdata = old_userdata
assert ( tree , " In file " .. tostring ( colored.blue ( filename ) ) .. " failed to parse: \n " .. tostring ( colored.onyellow ( colored.black ( nomsu_code ) ) ) )
2018-05-03 16:30:55 -07:00
if next ( userdata.errors ) then
local keys = utils.keys ( userdata.errors )
table.sort ( keys )
local errors
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # keys do
local k = keys [ _index_0 ]
_accum_0 [ _len_0 ] = userdata.errors [ k ]
_len_0 = _len_0 + 1
end
errors = _accum_0
end
error ( concat ( errors , " \n \n " ) , 0 )
end
2018-05-26 15:04:31 -07:00
return tree
2017-09-12 21:38:54 -07:00
end ,
2018-04-19 19:43:23 -07:00
run = function ( self , nomsu_code , compile_fn )
if compile_fn == nil then
compile_fn = nil
2018-04-18 15:28:46 -07:00
end
2018-05-09 13:34:33 -07:00
if # tostring ( nomsu_code ) == 0 then
2018-04-18 15:28:46 -07:00
return nil
2017-11-01 20:11:44 -07:00
end
2018-04-19 19:43:23 -07:00
local tree = self : parse ( nomsu_code )
2018-04-11 20:05:12 -07:00
assert ( tree , " Failed to parse: " .. tostring ( nomsu_code ) )
2018-05-16 19:08:16 -07:00
local lua = self : tree_to_lua ( tree ) : as_statements ( )
2018-04-13 14:54:35 -07:00
lua : declare_locals ( )
2018-04-19 19:43:23 -07:00
lua : prepend ( " -- File: " .. tostring ( nomsu_code.source or " " ) .. " \n " )
if compile_fn then
compile_fn ( lua )
end
2018-04-18 15:28:46 -07:00
return self : run_lua ( lua )
2018-01-08 18:53:57 -08:00
end ,
2018-04-28 17:08:28 -07:00
run_file = function ( self , filename , compile_fn )
2018-04-19 19:43:23 -07:00
if compile_fn == nil then
compile_fn = nil
end
2018-05-14 15:37:15 -07:00
local loaded = self.environment . LOADED
if loaded [ filename ] then
return loaded [ filename ]
end
2018-04-28 15:20:48 -07:00
local ret = nil
2018-04-28 17:08:28 -07:00
for filename in all_files ( filename ) do
2018-04-28 15:20:48 -07:00
local _continue_0 = false
repeat
2018-05-14 15:37:15 -07:00
if loaded [ filename ] then
ret = loaded [ filename ]
_continue_0 = true
break
end
for i , running in ipairs ( _running_files ) do
if running == filename then
local loop
do
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
end
loop = _accum_0
end
insert ( loop , filename )
error ( " Circular import, this loops forever: " .. tostring ( concat ( loop , " -> " ) ) )
end
end
insert ( _running_files , filename )
2018-04-28 15:20:48 -07:00
if filename : match ( " %.lua$ " ) then
local file = assert ( FILE_CACHE [ filename ] , " Could not find file: " .. tostring ( filename ) )
ret = self : run_lua ( Lua ( Source ( filename ) , file ) )
2018-04-28 18:07:14 -07:00
elseif filename : match ( " %.nom$ " ) or filename : match ( " ^/dev/fd/[012]$ " ) then
2018-04-28 15:20:48 -07:00
if not self.skip_precompiled then
local lua_filename = filename : gsub ( " %.nom$ " , " .lua " )
local file = FILE_CACHE [ lua_filename ]
if file then
ret = self : run_lua ( Lua ( Source ( filename ) , file ) )
2018-05-15 15:21:32 -07:00
remove ( _running_files )
2018-04-28 15:20:48 -07:00
_continue_0 = true
break
end
end
local file = file or FILE_CACHE [ filename ]
if not file then
error ( " File does not exist: " .. tostring ( filename ) , 0 )
end
ret = self : run ( Nomsu ( Source ( filename ) , file ) , compile_fn )
else
error ( " Invalid filetype for " .. tostring ( filename ) , 0 )
2018-01-10 16:22:45 -08:00
end
2018-05-14 15:37:15 -07:00
loaded [ filename ] = ret or true
remove ( _running_files )
2018-04-28 15:20:48 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
2018-01-10 16:22:45 -08:00
end
end
2018-05-14 15:37:15 -07:00
loaded [ filename ] = ret or true
2018-04-28 15:20:48 -07:00
return ret
2018-01-10 16:22:45 -08:00
end ,
2018-04-18 15:28:46 -07:00
run_lua = function ( self , lua )
assert ( type ( lua ) ~= ' string ' , " Attempt to run lua string instead of Lua (object) " )
2018-04-24 20:17:19 -07:00
local lua_string = tostring ( lua )
2018-04-18 15:28:46 -07:00
local run_lua_fn , err = load ( lua_string , filename , " t " , self.environment )
2018-01-12 16:33:11 -08:00
if not run_lua_fn then
2018-01-08 18:53:57 -08:00
local n = 1
local fn
fn = function ( )
n = n + 1
return ( " \n %-3d| " ) : format ( n )
end
2018-04-18 15:28:46 -07:00
local line_numbered_lua = " 1 | " .. lua_string : gsub ( " \n " , fn )
2018-04-11 20:05:12 -07:00
error ( " Failed to compile generated code: \n " .. tostring ( colored.bright ( colored.blue ( colored.onblack ( line_numbered_lua ) ) ) ) .. " \n \n " .. tostring ( err ) , 0 )
2018-01-08 18:53:57 -08:00
end
2018-01-12 16:33:11 -08:00
return run_lua_fn ( )
2017-09-24 20:20:43 -07:00
end ,
2018-05-26 15:04:31 -07:00
tree_to_lua = function ( self , tree )
2018-05-16 19:08:16 -07:00
local _exp_0 = tree.type
if " Action " == _exp_0 then
local stub = tree : get_stub ( )
local compile_action = self.environment . COMPILE_ACTIONS [ stub ]
if compile_action then
local args
do
local _accum_0 = { }
local _len_0 = 1
2018-05-26 15:04:31 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local arg = _list_0 [ _index_0 ]
2018-05-24 21:16:51 -07:00
if type ( arg ) ~= " string " then
2018-05-16 19:08:16 -07:00
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.environment . ARG_ORDERS [ compile_action ] [ stub ]
for _index_0 = 1 , # _list_0 do
local p = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p - 1 ]
_len_0 = _len_0 + 1
end
args = _accum_0
end
local ret = compile_action ( tree , unpack ( args ) )
if not ret then
error ( " Failed to produce any Lua " )
end
return ret
end
local action = rawget ( self.environment . ACTIONS , stub )
2018-05-26 15:04:31 -07:00
local lua = Lua.Value ( tree.source )
2018-05-16 19:08:16 -07:00
if not action and math_expression : match ( stub ) then
2018-05-26 15:04:31 -07:00
for i , tok in ipairs ( tree.value ) do
2018-05-24 21:16:51 -07:00
if type ( tok ) == ' string ' then
lua : append ( tok )
2018-05-16 19:08:16 -07:00
else
2018-05-26 15:04:31 -07:00
local tok_lua = self : tree_to_lua ( tok )
2018-05-16 19:08:16 -07:00
if not ( tok_lua.is_value ) then
error ( " non-expression value inside math expression: " .. tostring ( colored.yellow ( repr ( tok ) ) ) )
end
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
lua : append ( tok_lua )
end
2018-05-26 15:04:31 -07:00
if i < # tree.value then
2018-05-16 19:08:16 -07:00
lua : append ( " " )
end
end
return lua
end
local args = { }
2018-05-26 15:04:31 -07:00
for i , tok in ipairs ( tree.value ) do
2018-05-16 19:08:16 -07:00
local _continue_0 = false
repeat
2018-05-24 21:16:51 -07:00
if type ( tok ) == " string " then
2018-05-16 19:08:16 -07:00
_continue_0 = true
break
end
2018-05-26 15:04:31 -07:00
local arg_lua = self : tree_to_lua ( tok )
2018-05-16 19:08:16 -07:00
if not ( arg_lua.is_value ) then
error ( " Cannot use: \n " .. tostring ( colored.yellow ( repr ( tok ) ) ) .. " \n as an argument to " .. tostring ( stub ) .. " , since it's not an expression, it produces: " .. tostring ( repr ( arg_lua ) ) , 0 )
end
insert ( args , arg_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
if action then
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.environment . ARG_ORDERS [ action ] [ stub ]
for _index_0 = 1 , # _list_0 do
local p = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p ]
_len_0 = _len_0 + 1
end
args = _accum_0
end
end
lua : append ( " ACTIONS[ " , repr ( stub ) , " ]( " )
for i , arg in ipairs ( args ) do
lua : append ( arg )
if i < # args then
lua : append ( " , " )
end
end
lua : append ( " ) " )
return lua
elseif " EscapedNomsu " == _exp_0 then
local make_tree
make_tree = function ( t )
if type ( t ) ~= ' userdata ' then
return repr ( t )
end
if t.is_multi then
local bits
do
local _accum_0 = { }
local _len_0 = 1
2018-05-26 15:04:31 -07:00
local _list_0 = t.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
2018-05-16 19:08:16 -07:00
_accum_0 [ _len_0 ] = make_tree ( bit )
_len_0 = _len_0 + 1
end
bits = _accum_0
end
2018-05-26 15:04:31 -07:00
return t.type .. " (Tuple( " .. table.concat ( bits , " , " ) .. " ), " .. repr ( t.source ) .. " ) "
2018-05-16 19:08:16 -07:00
else
2018-05-26 15:04:31 -07:00
return t.type .. " ( " .. repr ( t.value ) .. " , " .. repr ( t.source ) .. " ) "
2018-05-16 19:08:16 -07:00
end
end
2018-05-26 15:04:31 -07:00
return Lua.Value ( tree.source , make_tree ( tree.value [ 1 ] ) )
2018-05-16 19:08:16 -07:00
elseif " Block " == _exp_0 then
2018-05-26 15:04:31 -07:00
local lua = Lua ( tree.source )
for i , line in ipairs ( tree.value ) do
local line_lua = self : tree_to_lua ( line )
2018-05-16 19:08:16 -07:00
if i > 1 then
lua : append ( " \n " )
end
lua : append ( line_lua : as_statements ( ) )
end
return lua
elseif " Text " == _exp_0 then
2018-05-26 15:04:31 -07:00
local lua = Lua.Value ( tree.source )
2018-05-16 19:08:16 -07:00
local string_buffer = " "
2018-05-26 15:04:31 -07:00
for i , bit in ipairs ( tree.value ) do
2018-05-16 19:08:16 -07:00
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
2018-05-26 15:04:31 -07:00
local bit_lua = self : tree_to_lua ( bit )
2018-05-16 19:08:16 -07:00
if not ( bit_lua.is_value ) then
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( bit ) ) ) .. " as a string interpolation value, since it's not an expression. " , 0 )
end
if # lua.bits > 0 then
lua : append ( " .. " )
end
if bit.type ~= " Text " then
2018-05-26 15:04:31 -07:00
bit_lua = Lua.Value ( bit.source , " stringify( " , bit_lua , " ) " )
2018-05-16 19:08:16 -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
lua : append ( repr ( string_buffer ) )
end
if # lua.bits > 1 then
lua : parenthesize ( )
end
return lua
elseif " List " == _exp_0 then
2018-05-26 15:04:31 -07:00
local lua = Lua.Value ( tree.source , " { " )
2018-05-16 19:08:16 -07:00
local line_length = 0
2018-05-26 15:04:31 -07:00
for i , item in ipairs ( tree.value ) do
local item_lua = self : tree_to_lua ( item )
2018-05-16 19:08:16 -07:00
if not ( item_lua.is_value ) then
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( item ) ) ) .. " as a list item, since it's not an expression. " , 0 )
end
lua : append ( item_lua )
local item_string = tostring ( item_lua )
local last_line = item_string : match ( " [^ \n ]*$ " )
if item_string : match ( " \n " ) then
line_length = # last_line
else
line_length = line_length + # last_line
end
2018-05-26 15:04:31 -07:00
if i < # tree.value then
2018-05-16 19:08:16 -07:00
if line_length >= MAX_LINE then
lua : append ( " , \n " )
line_length = 0
else
lua : append ( " , " )
line_length = line_length + 2
end
end
end
lua : append ( " } " )
return lua
elseif " Dict " == _exp_0 then
2018-05-26 15:04:31 -07:00
local lua = Lua.Value ( tree.source , " { " )
2018-05-16 19:08:16 -07:00
local line_length = 0
2018-05-26 15:04:31 -07:00
for i , entry in ipairs ( tree.value ) do
local entry_lua = self : tree_to_lua ( entry )
2018-05-16 19:08:16 -07:00
lua : append ( entry_lua )
local entry_lua_str = tostring ( entry_lua )
local last_line = entry_lua_str : match ( " \n ([^ \n ]*)$ " )
if last_line then
line_length = # last_line
else
line_length = line_length + # entry_lua_str
end
2018-05-26 15:04:31 -07:00
if i < # tree.value then
2018-05-16 19:08:16 -07:00
if line_length >= MAX_LINE then
lua : append ( " , \n " )
line_length = 0
else
lua : append ( " , " )
line_length = line_length + 2
end
end
end
lua : append ( " } " )
return lua
elseif " DictEntry " == _exp_0 then
2018-05-26 15:04:31 -07:00
local key , value = tree.value [ 1 ] , tree.value [ 2 ]
local key_lua = self : tree_to_lua ( key )
2018-05-16 19:08:16 -07:00
if not ( key_lua.is_value ) then
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( key ) ) ) .. " as a dict key, since it's not an expression. " , 0 )
end
2018-05-26 15:04:31 -07:00
local value_lua = value and self : tree_to_lua ( value ) or Lua.Value ( key.source , " true " )
2018-05-16 19:08:16 -07:00
if not ( value_lua.is_value ) then
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( value ) ) ) .. " as a dict value, since it's not an expression. " , 0 )
end
local key_str = tostring ( key_lua ) : match ( [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=] )
if key_str then
2018-05-26 15:04:31 -07:00
return Lua ( tree.source , key_str , " = " , value_lua )
2018-05-16 19:08:16 -07:00
elseif tostring ( key_lua ) : sub ( 1 , 1 ) == " [ " then
2018-05-26 15:04:31 -07:00
return Lua ( tree.source , " [ " , key_lua , " ]= " , value_lua )
2018-05-16 19:08:16 -07:00
else
2018-05-26 15:04:31 -07:00
return Lua ( tree.source , " [ " , key_lua , " ]= " , value_lua )
2018-05-16 19:08:16 -07:00
end
elseif " IndexChain " == _exp_0 then
2018-05-26 15:04:31 -07:00
local lua = self : tree_to_lua ( tree.value [ 1 ] )
2018-05-16 19:08:16 -07:00
if not ( lua.is_value ) then
2018-05-26 15:04:31 -07:00
error ( " Cannot index " .. tostring ( colored.yellow ( repr ( tree.value [ 1 ] ) ) ) .. " , since it's not an expression. " , 0 )
2018-05-16 19:08:16 -07:00
end
local first_char = tostring ( lua ) : sub ( 1 , 1 )
if first_char == " { " or first_char == ' " ' or first_char == " [ " then
lua : parenthesize ( )
end
2018-05-26 15:04:31 -07:00
for i = 2 , # tree.value do
local key = tree.value [ i ]
local key_lua = self : tree_to_lua ( key )
2018-05-16 19:08:16 -07:00
if not ( key_lua.is_value ) then
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( key ) ) ) .. " as an index, since it's not an expression. " , 0 )
end
local key_lua_str = tostring ( key_lua )
do
local lua_id = key_lua_str : match ( " ^[' \" ]([a-zA-Z_][a-zA-Z0-9_]*)[' \" ]$ " )
if lua_id then
lua : append ( " . " .. tostring ( lua_id ) )
elseif key_lua_str : sub ( 1 , 1 ) == ' [ ' then
lua : append ( " [ " , key_lua , " ] " )
else
lua : append ( " [ " , key_lua , " ] " )
end
end
end
return lua
elseif " Number " == _exp_0 then
2018-05-26 15:04:31 -07:00
return Lua.Value ( tree.source , tostring ( tree.value ) )
2018-05-16 19:08:16 -07:00
elseif " Var " == _exp_0 then
2018-05-26 15:04:31 -07:00
return Lua.Value ( tree.source , tree : as_lua_id ( ) )
2018-05-16 19:08:16 -07:00
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
end
end ,
tree_to_nomsu = function ( self , tree , inline , can_use_colon )
if inline == nil then
inline = false
end
if can_use_colon == nil then
can_use_colon = false
end
local _exp_0 = tree.type
if " Action " == _exp_0 then
if inline then
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source )
for i , bit in ipairs ( tree.value ) do
2018-05-24 21:16:51 -07:00
if type ( bit ) == " string " then
2018-05-16 19:08:16 -07:00
if i > 1 then
nomsu : append ( " " )
end
2018-05-24 21:16:51 -07:00
nomsu : append ( bit )
2018-05-16 19:08:16 -07:00
else
local arg_nomsu = self : tree_to_nomsu ( bit , true )
if not ( arg_nomsu ) then
return nil
end
if not ( i == 1 ) then
nomsu : append ( " " )
end
if bit.type == " Action " or bit.type == " Block " then
arg_nomsu : parenthesize ( )
end
nomsu : append ( arg_nomsu )
end
end
return nomsu
else
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source )
2018-05-16 19:08:16 -07:00
local next_space = " "
local last_colon = nil
2018-05-26 15:04:31 -07:00
for i , bit in ipairs ( tree.value ) do
2018-05-24 21:16:51 -07:00
if type ( bit ) == " string " then
nomsu : append ( next_space , bit )
2018-05-16 19:08:16 -07:00
next_space = " "
else
local arg_nomsu
if last_colon == i - 1 and bit.type == " Action " then
arg_nomsu = nil
elseif bit.type == " Block " then
arg_nomsu = nil
else
arg_nomsu = self : tree_to_nomsu ( bit , true )
end
if arg_nomsu and # arg_nomsu < MAX_LINE then
if bit.type == " Action " then
if can_use_colon and i > 1 then
nomsu : append ( next_space : match ( " [^ ]* " ) , " : " , arg_nomsu )
next_space = " \n .. "
last_colon = i
else
nomsu : append ( next_space , " ( " , arg_nomsu , " ) " )
next_space = " "
end
else
nomsu : append ( next_space , arg_nomsu )
next_space = " "
end
else
arg_nomsu = self : tree_to_nomsu ( bit , nil , true )
if not ( nomsu ) then
return nil
end
if bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
if i == 1 then
2018-05-26 15:04:31 -07:00
arg_nomsu = Nomsu ( bit.source , " (..) \n " , arg_nomsu )
2018-05-16 19:08:16 -07:00
else
2018-05-26 15:04:31 -07:00
arg_nomsu = Nomsu ( bit.source , " \n " , arg_nomsu )
2018-05-16 19:08:16 -07:00
end
end
if last_colon == i - 1 and ( bit.type == " Action " or bit.type == " Block " ) then
next_space = " "
end
nomsu : append ( next_space , arg_nomsu )
next_space = " \n .. "
end
if next_space == " " and # ( tostring ( nomsu ) : match ( " [^ \n ]*$ " ) ) > MAX_LINE then
next_space = " \n .. "
end
end
end
return nomsu
end
elseif " EscapedNomsu " == _exp_0 then
local nomsu = self : tree_to_nomsu ( tree.value , true )
if nomsu == nil and not inline then
2018-05-26 15:04:31 -07:00
nomsu = self : tree_to_nomsu ( tree.value [ 1 ] )
return nomsu and Nomsu ( tree.source , " \\ : \n " , nomsu )
2018-05-16 19:08:16 -07:00
end
2018-05-26 15:04:31 -07:00
return nomsu and Nomsu ( tree.source , " \\ ( " , nomsu , " ) " )
2018-05-16 19:08:16 -07:00
elseif " Block " == _exp_0 then
if inline then
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source )
for i , line in ipairs ( tree.value ) do
2018-05-16 19:08:16 -07:00
if i > 1 then
nomsu : append ( " ; " )
end
local line_nomsu = self : tree_to_nomsu ( line , true )
if not ( line_nomsu ) then
return nil
end
nomsu : append ( line_nomsu )
end
return nomsu
end
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source )
2018-05-16 19:08:16 -07:00
for i , line in ipairs ( self ) do
line = assert ( self : tree_to_nomsu ( line , nil , true ) , " Could not convert line to nomsu " )
nomsu : append ( line )
if i < # self then
nomsu : append ( " \n " )
if tostring ( line ) : match ( " \n " ) then
nomsu : append ( " \n " )
end
end
end
return nomsu
elseif " Text " == _exp_0 then
if inline then
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source , ' " ' )
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
2018-05-16 19:08:16 -07:00
if type ( bit ) == ' string ' then
nomsu : append ( ( bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \\ n " ) ) )
else
local interp_nomsu = self : tree_to_nomsu ( bit , true )
if interp_nomsu then
2018-05-24 21:16:51 -07:00
if bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
2018-05-16 19:08:16 -07:00
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
else
return nil
end
end
end
nomsu : append ( ' " ' )
return nomsu
else
local inline_version = self : tree_to_nomsu ( tree , true )
if inline_version and # inline_version <= MAX_LINE then
return inline_version
end
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source , ' ".." \n ' )
2018-05-16 19:08:16 -07:00
for i , bit in ipairs ( self ) do
if type ( bit ) == ' string ' then
nomsu : append ( ( bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \n " ) ) )
else
local interp_nomsu = self : tree_to_nomsu ( bit , true )
if interp_nomsu then
2018-05-24 21:16:51 -07:00
if bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
2018-05-16 19:08:16 -07:00
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
else
interp_nomsu = self : tree_to_nomsu ( bit )
if not ( interp_nomsu ) then
return nil
end
nomsu : append ( " \\ \n " , interp_nomsu )
if i < # self then
nomsu : append ( " \n .. " )
end
end
end
end
return nomsu
end
elseif " List " == _exp_0 then
if inline then
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source , " [ " )
for i , item in ipairs ( tree.value ) do
2018-05-16 19:08:16 -07:00
local item_nomsu = self : tree_to_nomsu ( item , true )
if not ( item_nomsu ) then
return nil
end
if i > 1 then
nomsu : append ( " , " )
end
nomsu : append ( item_nomsu )
end
nomsu : append ( " ] " )
return nomsu
else
local inline_version = self : tree_to_nomsu ( tree , true )
if inline_version and # inline_version <= MAX_LINE then
return inline_version
end
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source , " [..] " )
local line = Nomsu ( tree.source , " \n " )
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local item = _list_0 [ _index_0 ]
2018-05-16 19:08:16 -07:00
local item_nomsu = self : tree_to_nomsu ( item , true )
if item_nomsu and # line + # " , " + # item_nomsu <= MAX_LINE then
if # line.bits > 1 then
line : append ( " , " )
end
line : append ( item_nomsu )
else
if not ( item_nomsu ) then
item_nomsu = self : tree_to_nomsu ( item )
if not ( item_nomsu ) then
return nil
end
end
if # line.bits > 1 then
nomsu : append ( line )
2018-05-26 15:04:31 -07:00
line = Nomsu ( line.source , " \n " )
2018-05-16 19:08:16 -07:00
end
line : append ( item_nomsu )
end
end
if # line.bits > 1 then
nomsu : append ( line )
end
return nomsu
end
elseif " Dict " == _exp_0 then
if inline then
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source , " { " )
for i , entry in ipairs ( tree.value ) do
2018-05-16 19:08:16 -07:00
local entry_nomsu = self : tree_to_nomsu ( entry , true )
if not ( entry_nomsu ) then
return nil
end
if i > 1 then
nomsu : append ( " , " )
end
nomsu : append ( entry_nomsu )
end
nomsu : append ( " } " )
return nomsu
else
local inline_version = self : tree_to_nomsu ( tree , true )
if inline_version then
return inline_version
end
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source , " {..} " )
local line = Nomsu ( tree.source , " \n " )
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local entry = _list_0 [ _index_0 ]
2018-05-16 19:08:16 -07:00
local entry_nomsu = self : tree_to_nomsu ( entry )
if not ( entry_nomsu ) then
return nil
end
if # line + # tostring ( entry_nomsu ) <= MAX_LINE then
if # line.bits > 1 then
line : append ( " , " )
end
line : append ( entry_nomsu )
else
if # line.bits > 1 then
nomsu : append ( line )
2018-05-26 15:04:31 -07:00
line = Nomsu ( line.source , " \n " )
2018-05-16 19:08:16 -07:00
end
line : append ( entry_nomsu )
end
end
if # line.bits > 1 then
nomsu : append ( line )
end
return nomsu
end
elseif " DictEntry " == _exp_0 then
2018-05-26 15:04:31 -07:00
local key , value = tree.value [ 1 ] , tree.value [ 2 ]
2018-05-16 19:08:16 -07:00
local key_nomsu = self : tree_to_nomsu ( key , true )
if not ( key_nomsu ) then
return nil
end
if key.type == " Action " or key.type == " Block " then
key_nomsu : parenthesize ( )
end
local value_nomsu
if value then
value_nomsu = self : tree_to_nomsu ( value , true )
else
2018-05-26 15:04:31 -07:00
value_nomsu = Nomsu ( tree.source , " " )
2018-05-16 19:08:16 -07:00
end
if inline and not value_nomsu then
return nil
end
if not value_nomsu then
if inline then
return nil
end
value_nomsu = self : tree_to_nomsu ( value )
if not ( value_nomsu ) then
return nil
end
end
2018-05-26 15:04:31 -07:00
return Nomsu ( tree.source , key_nomsu , " : " , value_nomsu )
2018-05-16 19:08:16 -07:00
elseif " IndexChain " == _exp_0 then
2018-05-26 15:04:31 -07:00
local nomsu = Nomsu ( tree.source )
for i , bit in ipairs ( tree.value ) do
2018-05-16 19:08:16 -07:00
if i > 1 then
nomsu : append ( " . " )
end
local bit_nomsu = self : tree_to_nomsu ( bit , true )
if not ( bit_nomsu ) then
return nil
end
if bit.type == " Action " or bit.type == " Block " then
bit_nomsu : parenthesize ( )
end
nomsu : append ( bit_nomsu )
end
return nomsu
elseif " Number " == _exp_0 then
2018-05-26 15:04:31 -07:00
return Nomsu ( tree.source , tostring ( tree.value ) )
2018-05-16 19:08:16 -07:00
elseif " Var " == _exp_0 then
2018-05-26 15:04:31 -07:00
return Nomsu ( tree.source , " % " , tree.value )
2018-05-16 19:08:16 -07:00
elseif " Comment " == _exp_0 then
if inline then
return nil
end
2018-05-26 15:04:31 -07:00
return Nomsu ( tree.source , " # " , tree.value : gsub ( " \n " , " \n " ) )
2018-05-16 19:08:16 -07:00
else
return error ( " Unknown type: " .. tostring ( tree.type ) )
end
end ,
2018-04-11 20:05:12 -07:00
tree_to_value = function ( self , tree )
2018-05-16 18:12:56 -07:00
if tree.type == ' Text ' and # tree == 1 and type ( tree [ 1 ] ) == ' string ' then
return tree [ 1 ]
2018-01-25 17:34:49 -08:00
end
2018-05-26 15:04:31 -07:00
local lua = Lua ( tree.source , " return " , self : tree_to_lua ( tree ) , " ; " )
2018-04-18 17:41:40 -07:00
return self : run_lua ( lua )
2017-09-12 21:38:54 -07:00
end ,
2017-09-26 15:27:01 -07:00
walk_tree = function ( self , tree , depth )
if depth == nil then
depth = 0
2017-09-24 20:20:43 -07:00
end
2017-09-26 15:27:01 -07:00
coroutine.yield ( tree , depth )
2018-05-16 18:12:56 -07:00
if tree.is_multi then
2018-05-26 15:04:31 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
2018-05-24 20:27:08 -07:00
if Types.is_node ( v ) then
self : walk_tree ( v , depth + 1 )
end
2017-09-24 20:20:43 -07:00
end
2017-09-26 15:27:01 -07:00
end
end ,
2017-09-12 21:38:54 -07:00
initialize_core = function ( self )
2018-01-12 16:33:11 -08:00
local nomsu = self
2018-05-03 21:55:46 -07:00
self : define_compile_action ( " immediately %block " , function ( self , _block )
2018-05-16 19:08:16 -07:00
local lua = nomsu : tree_to_lua ( _block ) : as_statements ( )
2018-04-11 20:05:12 -07:00
lua : declare_locals ( )
nomsu : run_lua ( lua )
2018-05-26 15:04:31 -07:00
return Lua ( _block.source , " if IMMEDIATE then \n " , lua , " \n end " )
2018-04-13 14:54:35 -07:00
end )
2018-05-15 14:53:37 -07:00
local add_lua_string_bits
add_lua_string_bits = function ( lua , code )
2018-04-20 16:23:53 -07:00
if code.type ~= " Text " then
2018-05-16 19:08:16 -07:00
lua : append ( " , " , nomsu : tree_to_lua ( code ) )
2018-04-20 16:23:53 -07:00
return
2018-04-13 14:54:35 -07:00
end
2018-05-26 15:04:31 -07:00
local _list_0 = code.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
2018-04-13 14:54:35 -07:00
lua : append ( " , " )
if type ( bit ) == " string " then
lua : append ( repr ( bit ) )
else
2018-05-16 19:08:16 -07:00
local bit_lua = nomsu : tree_to_lua ( bit )
2018-04-13 14:54:35 -07:00
if not ( bit_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( bit ) ) ) .. " as a string interpolation value, since it's not an expression. " )
2018-04-13 14:54:35 -07:00
end
lua : append ( bit_lua )
end
end
2018-04-20 16:23:53 -07:00
end
2018-05-03 21:55:46 -07:00
self : define_compile_action ( " Lua %code " , function ( self , _code )
2018-05-26 15:04:31 -07:00
local lua = Lua.Value ( _code.source , " Lua( " , repr ( _code.source ) )
2018-05-15 14:53:37 -07:00
add_lua_string_bits ( lua , _code )
2018-04-20 16:23:53 -07:00
lua : append ( " ) " )
return lua
end )
2018-05-03 21:55:46 -07:00
self : define_compile_action ( " Lua value %code " , function ( self , _code )
2018-05-26 15:04:31 -07:00
local lua = Lua.Value ( _code.source , " Lua.Value( " , repr ( _code.source ) )
2018-05-15 14:53:37 -07:00
add_lua_string_bits ( lua , _code )
2018-04-13 14:54:35 -07:00
lua : append ( " ) " )
return lua
2018-04-11 20:05:12 -07:00
end )
2018-05-15 14:53:37 -07:00
local add_lua_bits
add_lua_bits = function ( lua , code )
2018-05-26 15:04:31 -07:00
local _list_0 = code.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
2017-10-19 18:16:48 -07:00
if type ( bit ) == " string " then
2018-04-11 20:05:12 -07:00
lua : append ( bit )
2017-10-19 18:16:48 -07:00
else
2018-05-16 19:08:16 -07:00
local bit_lua = nomsu : tree_to_lua ( bit )
2018-04-11 21:07:13 -07:00
if not ( bit_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( bit ) ) ) .. " as a string interpolation value, since it's not an expression. " , 0 )
2018-01-25 17:34:49 -08:00
end
2018-04-11 20:05:12 -07:00
lua : append ( bit_lua )
2017-10-19 18:16:48 -07:00
end
end
2018-04-11 20:05:12 -07:00
return lua
2018-05-15 14:53:37 -07:00
end
self : define_compile_action ( " lua> %code " , function ( self , _code )
if _code.type ~= " Text " then
2018-05-26 15:04:31 -07:00
return Lua ( self.source , " nomsu:run_lua( " , nomsu : tree_to_lua ( _code ) , " ); " )
2018-05-15 14:53:37 -07:00
end
2018-05-26 15:04:31 -07:00
return add_lua_bits ( Lua ( self.source ) , _code )
2018-01-08 18:53:57 -08:00
end )
2018-05-03 21:55:46 -07:00
self : define_compile_action ( " =lua %code " , function ( self , _code )
2018-04-11 20:05:12 -07:00
if _code.type ~= " Text " then
2018-05-26 15:04:31 -07:00
return Lua.Value ( self.source , " nomsu:run_lua( " , nomsu : tree_to_lua ( _code ) , " :as_statements('return ')) " )
2018-01-24 13:13:03 -08:00
end
2018-05-26 15:04:31 -07:00
return add_lua_bits ( Lua.Value ( self.source ) , _code )
2018-01-08 18:53:57 -08:00
end )
2018-05-03 21:55:46 -07:00
return self : define_compile_action ( " use %path " , function ( self , _path )
2018-04-28 17:08:28 -07:00
local path = nomsu : tree_to_value ( _path )
2018-05-14 15:37:15 -07:00
nomsu : run_file ( path )
2018-05-26 15:04:31 -07:00
return Lua ( _path.source , " nomsu:run_file( " .. tostring ( repr ( path ) ) .. " ); " )
2018-01-08 18:53:57 -08:00
end )
2017-09-12 21:38:54 -07:00
end
}
_base_0.__index = _base_0
_class_0 = setmetatable ( {
2018-01-19 17:28:40 -08:00
__init = function ( self )
2018-01-24 12:37:52 -08:00
local NaN_surrogate = { }
local nil_surrogate = { }
2018-01-11 03:32:12 -08:00
self.ids = setmetatable ( { } , {
__mode = " k " ,
__index = function ( self , key )
2018-01-24 12:37:52 -08:00
if key == nil then
return self [ nil_surrogate ]
elseif key ~= key then
return self [ NaN_surrogate ]
end
2018-01-11 03:32:12 -08:00
local id = new_uuid ( )
self [ key ] = id
return id
end
} )
2018-02-08 16:22:57 -08:00
self.file_metadata = setmetatable ( { } , {
__mode = " k "
} )
2018-01-12 16:33:11 -08:00
self.environment = {
nomsu = self ,
repr = repr ,
stringify = stringify ,
utils = utils ,
lpeg = lpeg ,
re = re ,
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 ,
bit32 = bit32 ,
rawlen = rawlen ,
table = table ,
assert = assert ,
dofile = dofile ,
loadstring = loadstring ,
type = type ,
select = select ,
debug = debug ,
math = math ,
io = io ,
pairs = pairs ,
load = load ,
ipairs = ipairs
}
2018-05-10 22:47:03 -07:00
if jit then
self.environment . len = function ( x )
do
local mt = getmetatable ( x )
if mt then
if mt.__len then
return mt.__len ( x )
end
end
end
return # x
end
else
self.environment . len = ( function ( x )
return # x
end )
end
2018-05-15 16:36:21 -07:00
self.environment . ipairs = function ( x )
if type ( x ) == ' function ' then
return coroutine.wrap ( x )
elseif type ( x ) == ' thread ' then
return coroutine.resume , x , nil
else
do
local mt = getmetatable ( x )
if mt then
if mt.__ipairs then
return mt.__ipairs ( x )
end
else
return _ipairs ( x )
end
end
end
end
self.environment . pairs = function ( x )
if type ( x ) == ' function ' then
return coroutine.wrap ( x )
elseif type ( x ) == ' thread ' then
return coroutine.resume , x , nil
else
do
local mt = getmetatable ( x )
if mt then
if mt.__pairs then
return mt.__pairs ( x )
end
else
return _pairs ( x )
end
end
end
end
2018-03-05 18:44:26 -08:00
for k , v in pairs ( Types ) do
self.environment [ k ] = v
end
self.environment . Tuple = Tuple
2018-04-12 18:01:51 -07:00
self.environment . Lua = Lua
2018-04-18 15:28:46 -07:00
self.environment . Nomsu = Nomsu
self.environment . Source = Source
2018-02-06 22:06:39 -08:00
self.environment . ACTIONS = setmetatable ( { } , {
2018-01-19 17:28:40 -08:00
__index = function ( self , key )
2018-05-03 21:55:46 -07:00
return function ( ... )
return error ( " Attempt to run undefined action: " .. tostring ( key ) , 0 )
end
2018-01-19 17:28:40 -08:00
end
} )
2018-05-03 22:33:44 -07:00
self.environment . COMPILE_ACTIONS = { }
2018-05-03 21:55:46 -07:00
self.environment . ARG_ORDERS = setmetatable ( { } , {
__mode = " k "
} )
2018-01-19 17:28:40 -08:00
self.environment . LOADED = { }
2018-02-13 15:17:45 -08:00
self.environment . Types = Types
2018-01-19 17:28:40 -08:00
return self : initialize_core ( )
2017-09-12 21:38:54 -07:00
end ,
__base = _base_0 ,
2017-09-13 16:22:04 -07:00
__name = " NomsuCompiler "
2017-09-12 21:38:54 -07:00
} , {
__index = _base_0 ,
__call = function ( cls , ... )
local _self_0 = setmetatable ( { } , _base_0 )
cls.__init ( _self_0 , ... )
return _self_0
end
} )
_base_0.__class = _class_0
local self = _class_0
2018-01-25 17:34:49 -08:00
stub_defs = {
space = ( P ( ' ' ) + P ( ' \n .. ' ) ) ^ 0 ,
2018-05-15 20:32:22 -07:00
word = ( NOMSU_DEFS.ident_char ^ 1 + NOMSU_DEFS.operator ) ,
2018-05-24 14:57:24 -07:00
varname = ( R ( ' az ' , ' AZ ' , ' 09 ' ) + P ( ' _ ' ) + NOMSU_DEFS.utf8_char + ( - P ( " ' " ) * NOMSU_DEFS.operator ) ) ^ 0
2018-01-25 17:34:49 -08:00
}
2018-05-03 21:55:46 -07:00
stub_pattern = re.compile ( [ = [ { ~ ( % space -> ' ' ) ( ( ' % ' ( % varname -> ' ' ) ) / % word ) ? ( ( % space -> ' ' ) ( ( ' % ' ( % varname -> ' ' ) ) / % word ) ) * ( % space -> ' ' ) ~ }
] = ] , stub_defs )
2018-01-25 17:34:49 -08:00
var_pattern = re.compile ( " {| %space ((('%' {%varname}) / %word) %space)+ |} " , stub_defs )
2018-05-03 21:55:46 -07:00
_nomsu_chunk_counter = 0
2018-05-14 15:37:15 -07:00
_running_files = { }
2018-05-16 19:08:16 -07:00
MAX_LINE = 80
math_expression = re.compile ( [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]] )
2017-09-13 16:22:04 -07:00
NomsuCompiler = _class_0
2017-09-12 21:38:54 -07:00
end
2018-04-08 18:11:44 -07:00
if arg and debug_getinfo ( 2 ) . func ~= require then
2017-10-08 20:41:05 -07:00
colors = require ( ' consolecolors ' )
2018-05-03 16:30:55 -07:00
local parser = re.compile ( [ [ args <- { | ( flag " ; " ) * { : inputs : { | ( { file } " ; " ) * | } : } { : nomsu_args : { | ( " --; " ( { [ ^ ; ] * } " ; " ) * ) ? | } : } " ; " ? | } ! .
2018-04-28 19:16:39 -07:00
flag <-
{ : interactive : ( " -i " -> true ) : }
/ { : optimized : ( " -O " -> true ) : }
/ { : format : ( " -f " -> true ) : }
/ { : syntax : ( " -s " -> true ) : }
/ { : print_file : " -p " " ; " { file } : }
/ { : output_file : " -o " " ; " { file } : }
/ { : help : ( ( " -h " / " --help " ) -> true ) : }
2018-04-28 17:08:28 -07:00
file <- " - " / [ ^ ; ] +
2017-10-08 18:23:48 -07:00
] ] , {
2018-04-28 19:16:39 -07:00
[ " true " ] = function ( )
return true
end
2017-10-08 18:23:48 -07:00
} )
local args = concat ( arg , " ; " ) .. " ; "
2018-04-28 19:16:39 -07:00
args = parser : match ( args )
if not args or args.help then
2018-04-28 17:08:28 -07:00
print ( [ = [ Nomsu Compiler
2018-05-03 16:30:55 -07:00
Usage : ( lua nomsu.lua | moon nomsu.moon ) [ - i ] [ - O ] [ - f ] [ - s ] [ --help] [-o output] [-p print_file] file1 file2... [-- nomsu args...]
2018-04-28 17:08:28 -07:00
OPTIONS
- i Run the compiler in interactive mode ( REPL )
- O Run the compiler in optimized mode ( use precompiled . lua versions of Nomsu files , when available )
- f Auto - format the given Nomsu file and print the result .
- s Check the program for syntax errors .
- h /-- help Print this message .
- o < file > Output the compiled Lua file to the given file ( use " - " to output to stdout ; if outputting to stdout and - p is not specified , - p will default to / dev / null )
- p < file > Print to the specified file instead of stdout .
2018-04-28 19:16:39 -07:00
< input > Input file can be " - " to use stdin .
2018-04-28 17:08:28 -07:00
] = ] )
2017-10-08 18:23:48 -07:00
os.exit ( )
2017-09-12 21:38:54 -07:00
end
2018-01-11 01:03:52 -08:00
local nomsu = NomsuCompiler ( )
2018-05-03 16:30:55 -07:00
nomsu.environment . arg = args.nomsu_args
2018-04-08 18:11:44 -07:00
local ok , to_lua = pcall ( function ( )
return require ( ' moonscript.base ' ) . to_lua
end )
if not ok then
to_lua = nil
end
local moonscript_line_tables = setmetatable ( { } , {
__index = function ( self , filename )
if not ( to_lua ) then
return nil
end
2018-04-11 20:05:12 -07:00
local _ , line_table = to_lua ( FILE_CACHE [ filename ] )
2018-04-08 18:11:44 -07:00
self [ filename ] = line_table
return line_table
end
} )
2018-04-18 17:41:40 -07:00
debug.getinfo = function ( thread , f , what )
2018-04-12 18:01:51 -07:00
if what == nil then
f , what , thread = thread , f , nil
end
if type ( f ) == ' number ' then
f = f + 1
end
local info
if thread == nil then
info = debug_getinfo ( f , what )
else
info = debug_getinfo ( thread , f , what )
end
2018-04-08 18:11:44 -07:00
if not info or not info.func then
return info
end
2018-04-12 20:39:17 -07:00
if info.short_src or info.source or info.linedefine or info.currentline then
2018-05-03 21:55:46 -07:00
for k , v in pairs ( nomsu.environment . ACTIONS ) do
if v == info.func then
info.name = k
break
end
end
local _ = [ = [ if metadata = nomsu.action_metadata [ info.func ]
info.name = metadata.aliases [ 1 ]
filename = if type ( metadata.source ) == ' string '
2018-04-18 17:41:40 -07:00
metadata.source \ match ( " ^[^[:]* " )
else metadata.source . filename
info.short_src = filename
info.source = FILE_CACHE [ filename ]
ok , linedefined = pcall ( lua_line_to_nomsu_line , info.short_src , info.linedefined )
if ok then info.linedefined = linedefined
ok , currentline = pcall ( lua_line_to_nomsu_line , info.short_src , info.currentline )
--if ok then info.currentline = currentline
] = ]
2018-04-08 18:11:44 -07:00
end
return info
end
2018-04-28 18:07:14 -07:00
local print_err_msg
print_err_msg = function ( error_message , stack_offset )
if stack_offset == nil then
stack_offset = 2
2017-09-12 21:38:54 -07:00
end
2018-04-28 18:07:14 -07:00
io.stderr : write ( tostring ( colored.red ( " ERROR: " ) ) .. " " .. tostring ( colored.bright ( colored.red ( ( error_message or " " ) ) ) ) .. " \n " )
io.stderr : write ( " stack traceback: \n " )
2018-04-08 18:11:44 -07:00
ok , to_lua = pcall ( function ( )
2018-01-27 16:39:56 -08:00
return require ( ' moonscript.base ' ) . to_lua
end )
if not ok then
to_lua = function ( )
return nil
end
end
2018-04-20 16:23:53 -07:00
local nomsu_source = FILE_CACHE [ " nomsu.moon " ]
2018-01-11 01:03:52 -08:00
local _ , line_table = to_lua ( nomsu_source )
2018-04-28 18:07:14 -07:00
local level = stack_offset
2017-09-14 15:35:06 -07:00
while true do
2018-01-11 01:03:52 -08:00
local _continue_0 = false
repeat
2018-04-08 18:11:44 -07:00
local calling_fn = debug_getinfo ( level )
2018-01-11 01:03:52 -08:00
if not calling_fn then
2017-10-08 18:23:48 -07:00
break
end
2018-01-11 01:03:52 -08:00
if calling_fn.func == run then
break
end
level = level + 1
local name = calling_fn.name
if name == " run_lua_fn " then
_continue_0 = true
break
end
local line = nil
2018-05-09 13:34:33 -07:00
_ = [ = [ if metadata = nomsu.action_metadata [ calling_fn.func ]
filename , start , stop = metadata.source \ match ( " ([^:]*):([0-9]*),([0-9]*) " )
if filename
file = FILE_CACHE [ filename ]
line_no = 1
for _ in file \ sub ( 1 , tonumber ( start ) ) \ gmatch ( " \n " ) do line_no += 1
offending_statement = file \ sub ( tonumber ( start ) , tonumber ( stop ) )
if # offending_statement > 50
offending_statement = offending_statement \ sub ( 1 , 50 ) .. " ... "
offending_statement = colored.red ( offending_statement )
line = colored.yellow ( filename .. " : " .. tostring ( line_no ) .. " \n " .. offending_statement )
else
line = colored.yellow ( metadata.source )
name = colored.bright ( colored.yellow ( metadata.aliases [ 1 ] ) )
2018-01-25 17:34:49 -08:00
else
2018-05-09 13:34:33 -07:00
if calling_fn.istailcall and not name
name = " <tail call> "
if calling_fn.short_src == " ./nomsu.moon " and line_table
char = line_table [ calling_fn.currentline ]
line_num = 1
for _ in nomsu_source \ sub ( 1 , char ) \ gmatch ( " \n " ) do line_num += 1
line = colored.cyan ( " #{calling_fn.short_src}:#{line_num} " )
name = colored.bright ( colored.cyan ( name or " ??? " ) )
else
line = colored.blue ( " #{calling_fn.short_src}:#{calling_fn.currentline} " )
name = colored.bright ( colored.blue ( name or " ??? " ) )
_from = colored.dim colored.white " | "
io.stderr \ write ( ( " %32s %s %s \n " ) \ format ( name , _from , line ) )
] = ]
2018-01-11 01:03:52 -08:00
_continue_0 = true
until true
if not _continue_0 then
2017-09-14 15:35:06 -07:00
break
end
2017-09-14 18:18:42 -07:00
end
2018-04-28 18:07:14 -07:00
return io.stderr : flush ( )
end
local run
run = function ( )
2018-04-28 19:16:39 -07:00
for i , input in ipairs ( args.inputs ) do
if input == " - " then
args.inputs [ i ] = STDIN
end
end
if # args.inputs == 0 and not args.interactive then
args.inputs = {
" core "
}
args.interactive = true
2018-04-28 18:07:14 -07:00
end
local output_file
2018-04-28 19:16:39 -07:00
if args.output_file == " - " then
2018-04-28 18:07:14 -07:00
output_file = io.stdout
2018-04-28 19:16:39 -07:00
elseif args.output_file then
output_file = io.open ( args.output_file , ' w ' )
2018-04-28 18:07:14 -07:00
end
local print_file
if args.print_file == " - " then
print_file = io.stdout
elseif args.print_file then
print_file = io.open ( args.print_file , ' w ' )
elseif output_file == io.stdout then
print_file = nil
else
print_file = io.stdout
end
2018-04-28 19:16:39 -07:00
nomsu.skip_precompiled = not args.optimized
if print_file == nil then
nomsu.environment . print = function ( ) end
elseif print_file ~= io.stdout then
nomsu.environment . print = function ( ... )
local N = select ( " # " , ... )
if N > 0 then
print_file : write ( tostring ( select ( 1 , ... ) ) )
for i = 2 , N do
print_file : write ( ' \t ' , tostring ( select ( 1 , ... ) ) )
2018-04-28 18:07:14 -07:00
end
end
2018-04-28 19:16:39 -07:00
print_file : write ( ' \n ' )
return print_file : flush ( )
2018-04-28 18:07:14 -07:00
end
2018-04-28 19:16:39 -07:00
end
2018-05-03 21:55:46 -07:00
local compile_fn
2018-04-28 19:16:39 -07:00
if output_file then
compile_fn = function ( code )
output_file : write ( " local IMMEDIATE = true; \n " .. tostring ( code ) )
return output_file : flush ( )
2018-04-28 18:07:14 -07:00
end
2018-05-03 21:55:46 -07:00
else
compile_fn = nil
2018-04-28 19:16:39 -07:00
end
2018-05-03 16:30:55 -07:00
local parse_errs = { }
2018-04-28 19:16:39 -07:00
local _list_0 = args.inputs
for _index_0 = 1 , # _list_0 do
local input = _list_0 [ _index_0 ]
if args.syntax then
for input_file in all_files ( input ) do
2018-05-03 16:30:55 -07:00
local err
ok , err = pcall ( nomsu.parse , nomsu , Nomsu ( input_file , io.open ( input_file ) : read ( " *a " ) ) )
if not ok then
insert ( parse_errs , err )
elseif print_file then
print_file : write ( " Parse succeeded: " .. tostring ( input_file ) .. " \n " )
print_file : flush ( )
end
2018-04-28 18:07:14 -07:00
end
2018-04-28 19:16:39 -07:00
elseif args.format then
for input_file in all_files ( input ) do
2018-04-28 18:07:14 -07:00
local tree = nomsu : parse ( io.open ( input_file ) : read ( " *a " ) )
2018-05-16 19:08:16 -07:00
local formatted = tostring ( self : tree_to_nomsu ( tree ) )
2018-04-28 18:07:14 -07:00
if output_file then
output_file : write ( formatted , " \n " )
output_file : flush ( )
end
if print_file then
print_file : write ( formatted , " \n " )
print_file : flush ( )
end
end
2018-04-28 19:16:39 -07:00
elseif input == STDIN then
2018-04-28 18:07:14 -07:00
nomsu : run ( io.input ( ) : read ( " *a " ) , compile_fn )
else
2018-04-28 19:16:39 -07:00
nomsu : run_file ( input , compile_fn )
2018-04-28 18:07:14 -07:00
end
end
2018-05-03 16:30:55 -07:00
if # parse_errs > 0 then
io.stderr : write ( concat ( parse_errs , " \n \n " ) )
io.stderr : flush ( )
os.exit ( false , true )
elseif args.syntax then
os.exit ( true , true )
end
2018-04-28 19:16:39 -07:00
if args.interactive then
2018-04-28 18:07:14 -07:00
while true do
io.write ( colored.bright ( colored.yellow ( " >> " ) ) )
local buff = " "
while true do
local line = io.read ( " *L " )
if line == " \n " or not line then
if # buff > 0 then
io.write ( " \027 [1A \027 [2K " )
end
break
end
line = line : gsub ( " \t " , " " )
buff = buff .. line
io.write ( colored.dim ( colored.yellow ( " .. " ) ) )
end
if # buff == 0 then
break
end
local ret
ok , ret = pcall ( nomsu.run , nomsu , buff )
if ok and ret ~= nil then
print ( " = " .. repr ( ret ) )
elseif not ok then
print_err_msg ( ret )
end
end
end
end
local err_hand
err_hand = function ( error_message )
print_err_msg ( error_message )
2018-01-11 01:03:52 -08:00
return os.exit ( false , true )
2017-09-14 15:35:06 -07:00
end
2018-05-14 14:45:38 -07:00
do
local ldt
ok , ldt = pcall ( require , ' ldt ' )
if ok then
ldt.guard ( run )
else
xpcall ( run , err_hand )
end
2018-04-28 18:07:14 -07:00
end
2017-09-12 21:38:54 -07:00
end
2017-09-13 16:22:04 -07:00
return NomsuCompiler