2017-09-12 21:38:54 -07:00
local re = require ( ' re ' )
local lpeg = require ( ' lpeg ' )
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 ' )
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
} )
2017-09-28 17:49:15 -07:00
local colored = setmetatable ( { } , {
__index = function ( _ , color )
return ( function ( msg )
2017-12-30 14:31:07 -08:00
return colors [ color ] .. ( 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-01-23 19:22:20 -08:00
do
local STRING_METATABLE = getmetatable ( " " )
STRING_METATABLE.__add = function ( self , other )
return self .. stringify ( other )
end
end
2017-09-12 21:38:54 -07:00
lpeg.setmaxstack ( 10000 )
2017-12-08 15:37:36 -08:00
local P , R , V , S , Cg , C , Cp , B , Cmt
P , R , V , S , Cg , C , Cp , B , Cmt = lpeg.P , lpeg.R , lpeg.V , lpeg.S , lpeg.Cg , lpeg.C , lpeg.Cp , lpeg.B , lpeg.Cmt
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 = { }
_with_0.nl = P ( " \n " )
_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 )
2017-12-30 14:31:07 -08:00
local spaces = self : match ( " [ \t ]* " , start )
2018-01-19 17:28:40 -08:00
if # spaces > lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ] then
insert ( lpeg.userdata . indent_stack , # spaces )
2017-12-30 14:31:07 -08:00
return start + # spaces
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 )
2017-12-30 14:31:07 -08:00
local spaces = self : match ( " [ \t ]* " , start )
2018-01-19 17:28:40 -08:00
if # spaces < lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ] then
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 )
2017-12-30 14:31:07 -08:00
local spaces = self : match ( " [ \t ]* " , start )
2018-01-19 17:28:40 -08:00
if # spaces == lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ] then
2017-12-30 14:31:07 -08:00
return start + # spaces
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.gt_nodent = P ( function ( self , start )
2017-12-30 14:31:07 -08:00
local spaces = self : match ( " [ \t ]* " , start )
2018-01-19 17:28:40 -08:00
if # spaces >= lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ] + 4 then
return start + lpeg.userdata . indent_stack [ # lpeg.userdata . indent_stack ] + 4
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.error = function ( src , pos , err_msg )
if lpeg.userdata . source_code : sub ( pos , pos ) == " \n " then
pos = pos + # lpeg.userdata . source_code : match ( " [ \t \n ]* " , pos )
end
local line_no = 1
while ( lpeg.userdata . line_starts [ line_no + 1 ] or math.huge ) < pos do
line_no = line_no + 1
end
local prev_line
if line_no > 1 then
prev_line = lpeg.userdata . source_code : match ( " [^ \n ]* " , lpeg.userdata . line_starts [ line_no - 1 ] )
else
prev_line = " "
end
local err_line = lpeg.userdata . source_code : match ( " [^ \n ]* " , lpeg.userdata . line_starts [ line_no ] )
local next_line
if line_no < # lpeg.userdata . line_starts then
next_line = lpeg.userdata . source_code : match ( " [^ \n ]* " , lpeg.userdata . line_starts [ line_no + 1 ] )
else
next_line = " "
end
local pointer = ( " - " ) : rep ( pos - lpeg.userdata . line_starts [ line_no ] ) .. " ^ "
err_msg = ( err_msg or " Parse error " ) .. " in " .. tostring ( lpeg.userdata . filename ) .. " on line " .. tostring ( line_no ) .. " : \n "
err_msg = err_msg .. " \n " .. tostring ( prev_line ) .. " \n " .. tostring ( err_line ) .. " \n " .. tostring ( pointer ) .. " \n " .. tostring ( next_line ) .. " \n "
return error ( err_msg )
end
_with_0.FunctionCall = function ( start , value , stop )
local stub = concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # value do
local t = value [ _index_0 ]
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " " )
local src = lpeg.userdata . source_code : sub ( start , stop - 1 )
return {
start = start ,
stop = stop ,
type = " FunctionCall " ,
src = src ,
get_line_no = lpeg.userdata . get_line_no ,
value = value ,
stub = stub
}
end
NOMSU_DEFS = _with_0
end
setmetatable ( NOMSU_DEFS , {
__index = function ( self , key )
local make_node
make_node = function ( start , value , stop )
2017-12-30 14:31:07 -08:00
return {
2018-01-10 20:45:03 -08:00
start = start ,
stop = stop ,
2017-12-30 14:31:07 -08:00
value = value ,
2018-01-19 17:28:40 -08:00
src = lpeg.userdata . source_code : sub ( start , stop - 1 ) ,
get_line_no = lpeg.userdata . get_line_no ,
type = key
2017-12-30 14:31:07 -08:00
}
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
} )
local NOMSU
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 } " ) " ( " " * ) " : "
{ ( ( % nl " " + [ ^% nl ] * ) + ) / ( [ ^% nl ] * ) } ) -> " %1 <- ({} %3 {}) -> %2 "
ident <- [ a - zA - Z_ ] [ a - zA - Z0 - 9 _ ] *
comment <- " -- " [ ^% nl ] *
] ] )
2018-01-19 17:28:40 -08:00
local nomsu_peg = peg_tidier : match ( io.open ( " nomsu.peg " ) : read ( " *a " ) )
NOMSU = 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
local _base_0 = {
2017-09-14 21:03:42 -07:00
writeln = function ( self , ... )
self : write ( ... )
return self : write ( " \n " )
end ,
2017-10-12 14:39:49 -07:00
errorln = function ( self , ... )
self : write_err ( ... )
return self : write_err ( " \n " )
end ,
2018-01-11 02:07:37 -08:00
define_action = function ( self , signature , line_no , fn , src , compile_time )
if compile_time == nil then
compile_time = false
2017-09-12 21:38:54 -07:00
end
2017-10-13 16:10:47 -07:00
if type ( signature ) == ' string ' then
signature = self : get_stubs ( {
signature
} )
elseif type ( signature ) == ' table ' and type ( signature [ 1 ] ) == ' string ' then
signature = self : get_stubs ( signature )
end
2018-01-12 19:27:59 -08:00
assert ( type ( fn ) == ' function ' , " Bad fn: " .. tostring ( repr ( fn ) ) )
2017-10-08 15:06:05 -07:00
local aliases = { }
2017-12-04 17:35:47 -08:00
self.__class . def_number = self.__class . def_number + 1
2018-01-12 19:27:59 -08:00
local fn_info = debug.getinfo ( fn , " u " )
2018-01-18 01:49:13 -08:00
local fn_arg_positions , arg_orders
if not ( fn_info.isvararg ) then
do
local _tbl_0 = { }
for i = 1 , fn_info.nparams do
_tbl_0 [ debug.getlocal ( fn , i ) ] = i
end
fn_arg_positions = _tbl_0
2017-10-02 19:00:58 -07:00
end
2018-01-18 01:49:13 -08:00
arg_orders = { }
2018-01-12 19:27:59 -08:00
end
for sig_i = 1 , # signature do
local stub , arg_names = unpack ( signature [ sig_i ] )
assert ( stub , " NO STUB FOUND: " .. tostring ( repr ( signature ) ) )
if self.debug then
2018-01-18 01:49:13 -08:00
self : writeln ( tostring ( colored.bright ( " DEFINING ACTION: " ) ) .. " " .. tostring ( colored.underscore ( colored.magenta ( repr ( stub ) ) ) ) .. " " .. tostring ( colored.bright ( " WITH ARGS " ) ) .. " " .. tostring ( colored.dim ( repr ( arg_names ) ) ) .. " ON: " .. tostring ( self.environment . ACTIONS ) )
end
self.environment . ACTIONS [ stub ] = fn
if not ( fn_info.isvararg ) then
local arg_positions
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # arg_names do
local a = arg_names [ _index_0 ]
_accum_0 [ _len_0 ] = fn_arg_positions [ self : var_to_lua_identifier ( a ) ]
_len_0 = _len_0 + 1
end
arg_positions = _accum_0
end
assert ( # arg_positions == # arg_names , " Mismatch in args between lua function's " .. tostring ( repr ( fn_arg_positions ) ) .. " and stub's " .. tostring ( repr ( arg_names ) ) )
arg_orders [ stub ] = arg_positions
2017-12-04 17:35:47 -08:00
end
2017-09-18 22:41:50 -07:00
end
2018-01-12 19:27:59 -08:00
self.action_metadata [ fn ] = {
fn = fn ,
src = src ,
line_no = line_no ,
aliases = aliases ,
arg_orders = arg_orders ,
2018-01-18 01:49:13 -08:00
arg_positions = fn_arg_positions ,
2018-01-12 19:27:59 -08:00
def_number = self.__class . def_number
}
2017-09-21 21:11:13 -07:00
end ,
2018-01-11 02:07:37 -08:00
define_compile_action = function ( self , signature , line_no , fn , src )
2018-01-12 19:27:59 -08:00
self : define_action ( signature , line_no , fn , src , true )
self.action_metadata [ fn ] . compile_time = true
2018-01-18 01:49:13 -08:00
if self.debug then
return self : writeln ( tostring ( colored.bright ( colored.green ( " (it was compile time) " ) ) ) )
end
2017-12-04 17:35:47 -08:00
end ,
serialize_defs = function ( self , scope , after )
if scope == nil then
scope = nil
end
2017-10-31 16:19:08 -07:00
if after == nil then
2017-12-15 15:30:05 -08:00
after = nil
2017-10-31 16:19:08 -07:00
end
2018-01-12 19:27:59 -08:00
error ( " Not currently functional. " )
2017-12-15 15:30:05 -08:00
after = after or ( self.core_defs or 0 )
2017-12-04 17:35:47 -08:00
scope = scope or self.defs
local defs_by_num = { }
for stub , def in pairs ( scope ) do
if def and stub : sub ( 1 , 1 ) ~= " # " then
defs_by_num [ def.def_number ] = def
end
end
local keys
do
local _accum_0 = { }
local _len_0 = 1
for k , v in pairs ( defs_by_num ) do
_accum_0 [ _len_0 ] = k
_len_0 = _len_0 + 1
end
keys = _accum_0
2017-10-31 16:19:08 -07:00
end
table.sort ( keys )
local buff = { }
2017-12-04 17:35:47 -08:00
local k_i = 1
local _using = nil
local _using_do = { }
for k_i , i in ipairs ( keys ) do
local _continue_0 = false
repeat
if i <= after then
_continue_0 = true
break
end
local def = defs_by_num [ i ]
if def.defs == scope then
if def.src then
insert ( buff , def.src )
end
_continue_0 = true
break
end
if _using == def.defs then
if def.src then
insert ( _using_do , def.src )
end
else
_using = def.defs
_using_do = {
def.src
}
end
if k_i == # keys or defs_by_num [ keys [ k_i + 1 ] ] . defs ~= _using then
insert ( buff , " using: \n " .. tostring ( self : indent ( self : serialize_defs ( _using ) ) ) .. " \n ..do: \n " .. tostring ( self : indent ( concat ( _using_do , " \n " ) ) ) )
end
_continue_0 = true
until true
if not _continue_0 then
break
2017-10-31 16:19:08 -07:00
end
end
2017-12-04 17:35:47 -08:00
for k , v in pairs ( scope [ " #vars " ] or { } ) do
2017-12-14 14:26:24 -08:00
insert ( buff , " <% " .. tostring ( k ) .. " > = " .. tostring ( self : value_to_nomsu ( v ) ) )
2017-12-04 17:35:47 -08:00
end
2017-10-31 16:19:08 -07:00
return concat ( buff , " \n " )
end ,
2017-12-04 17:35:47 -08:00
dedent = function ( self , code )
if not ( code : find ( " \n " ) ) then
return code
end
local spaces , indent_spaces = math.huge , math.huge
for line in code : gmatch ( " \n ([^ \n ]*) " ) do
local _continue_0 = false
repeat
if line : match ( " ^%s*#.* " ) then
_continue_0 = true
break
else
do
local s = line : match ( " ^(%s*)%.%..* " )
if s then
spaces = math.min ( spaces , # s )
else
do
s = line : match ( " ^(%s*)%S.* " )
if s then
indent_spaces = math.min ( indent_spaces , # s )
end
end
end
end
end
_continue_0 = true
until true
if not _continue_0 then
break
end
end
if spaces ~= math.huge and spaces < indent_spaces then
return ( code : gsub ( " \n " .. ( " " ) : rep ( spaces ) , " \n " ) )
else
return ( code : gsub ( " \n " .. ( " " ) : rep ( indent_spaces ) , " \n " ) )
end
end ,
2018-01-08 18:53:57 -08:00
indent = function ( self , code , levels )
if levels == nil then
levels = 1
end
return code : gsub ( " \n " , " \n " .. ( " " ) : rep ( levels ) )
2017-12-04 17:35:47 -08:00
end ,
2018-01-19 17:28:40 -08:00
parse = function ( self , nomsu_code , filename )
2018-01-12 19:27:59 -08:00
assert ( type ( filename ) == " string " , " Bad filename type: " .. tostring ( type ( filename ) ) )
2017-09-12 21:38:54 -07:00
if self.debug then
2018-01-19 17:28:40 -08:00
self : writeln ( tostring ( colored.bright ( " PARSING: " ) ) .. " \n " .. tostring ( colored.yellow ( nomsu_code ) ) )
end
nomsu_code = nomsu_code : gsub ( " \r " , " " )
local userdata
do
local _with_0 = {
source_code = nomsu_code ,
filename = filename ,
indent_stack = {
0
}
}
_with_0.line_starts = re.compile ( " lines <- {| line (' \n ' line)* |} line <- {} [^ \n ]* " ) : match ( nomsu_code )
_with_0.get_line_no = function ( self )
if not ( self._line_no ) then
local line_no = 1
while ( _with_0.line_starts [ line_no + 1 ] or math.huge ) < self.start do
line_no = line_no + 1
end
self._line_no = tostring ( _with_0.filename ) .. " : " .. tostring ( line_no )
end
return self._line_no
end
userdata = _with_0
2017-09-12 21:38:54 -07:00
end
2018-01-19 17:28:40 -08:00
local old_userdata
old_userdata , lpeg.userdata = lpeg.userdata , userdata
local tree = NOMSU : match ( nomsu_code )
lpeg.userdata = old_userdata
assert ( tree , " In file " .. tostring ( colored.blue ( filename ) ) .. " failed to parse: \n " .. tostring ( colored.onyellow ( colored.black ( nomsu_code ) ) ) )
2017-09-12 21:38:54 -07:00
if self.debug then
2017-09-24 20:20:43 -07:00
self : writeln ( " PARSE TREE: " )
self : print_tree ( tree , " " )
2017-09-12 21:38:54 -07:00
end
return tree
end ,
2018-01-10 20:45:03 -08:00
run = function ( self , src , filename , max_operations , output_file )
2017-10-09 04:31:41 -07:00
if max_operations == nil then
max_operations = nil
end
2017-12-14 16:41:22 -08:00
if output_file == nil then
output_file = nil
end
2017-11-01 20:11:44 -07:00
if src == " " then
2018-01-10 20:45:03 -08:00
return nil , " "
2017-11-01 20:11:44 -07:00
end
2017-10-09 04:31:41 -07:00
if max_operations then
local timeout
timeout = function ( )
2017-10-09 04:37:16 -07:00
debug.sethook ( )
2018-01-12 19:27:59 -08:00
return error ( " Execution quota exceeded. Your code took too long. " )
2017-10-09 04:31:41 -07:00
end
debug.sethook ( timeout , " " , max_operations )
end
2017-09-24 20:20:43 -07:00
local tree = self : parse ( src , filename )
2018-01-12 19:27:59 -08:00
assert ( tree , " Failed to parse: " .. tostring ( src ) )
assert ( tree.type == " File " , " Attempt to run non-file: " .. tostring ( tree.type ) )
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( tree )
2018-01-08 18:53:57 -08:00
local lua_code = lua.statements or ( lua.expr .. " ; " )
2018-01-23 19:22:20 -08:00
local locals = lua_code.locals or { }
if # locals > 0 then
lua_code = " local " .. concat ( locals , " , " ) .. " ; \n " .. lua_code
end
2018-01-09 14:59:06 -08:00
lua_code = " -- File: " .. tostring ( filename ) .. " \n " .. lua_code
2018-01-10 20:45:03 -08:00
local ret = self : run_lua ( lua_code )
2017-10-10 00:52:07 -07:00
if max_operations then
debug.sethook ( )
end
2018-01-08 18:53:57 -08:00
if output_file then
output_file : write ( lua_code )
end
2018-01-10 20:45:03 -08:00
return ret , lua_code
2018-01-08 18:53:57 -08:00
end ,
2018-01-10 20:45:03 -08:00
run_file = function ( self , filename )
2018-01-10 16:22:45 -08:00
if filename : match ( " .*%.lua " ) then
2018-01-12 16:33:11 -08:00
local file = io.open ( filename )
local contents = file : read ( " *a " )
file : close ( )
return assert ( load ( contents , nil , nil , self.environment ) ) ( )
2018-01-10 16:22:45 -08:00
end
if filename : match ( " .*%.nom " ) then
if not self.skip_precompiled then
local file = io.open ( filename : gsub ( " %.nom " , " .lua " ) , " r " )
if file then
local lua_code = file : read ( " *a " )
file : close ( )
2018-01-10 20:45:03 -08:00
return self : run_lua ( lua_code )
2018-01-10 16:22:45 -08:00
end
end
local file = file or io.open ( filename )
if not file then
2018-01-12 19:27:59 -08:00
error ( " File does not exist: " .. tostring ( filename ) )
2018-01-10 16:22:45 -08:00
end
local nomsu_code = file : read ( ' *a ' )
file : close ( )
return self : run ( nomsu_code , filename )
else
2018-01-12 19:27:59 -08:00
return error ( " Invalid filetype for " .. tostring ( filename ) )
2018-01-10 16:22:45 -08:00
end
end ,
2018-01-10 20:45:03 -08:00
require_file = function ( self , filename )
2018-01-12 19:27:59 -08:00
local loaded = self.environment . LOADED
2018-01-10 16:22:45 -08:00
if not loaded [ filename ] then
2018-01-10 20:45:03 -08:00
loaded [ filename ] = self : run_file ( filename ) or true
2018-01-10 16:22:45 -08:00
end
return loaded [ filename ]
end ,
2018-01-10 20:45:03 -08:00
run_lua = function ( self , lua_code )
2018-01-12 16:33:11 -08:00
local run_lua_fn , err = load ( lua_code , nil , nil , self.environment )
2018-01-10 20:45:03 -08:00
if self.debug then
self : writeln ( tostring ( colored.bright ( " RUNNING LUA: " ) ) .. " \n " .. tostring ( colored.blue ( colored.bright ( lua_code ) ) ) )
end
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
local code = " 1 | " .. lua_code : gsub ( " \n " , fn )
2018-01-12 19:27:59 -08:00
error ( " Failed to compile generated code: \n " .. tostring ( colored.bright ( colored.blue ( colored.onblack ( code ) ) ) ) .. " \n \n " .. tostring ( err ) )
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-01-10 20:45:03 -08:00
tree_to_value = function ( self , tree , filename )
2018-01-12 16:33:11 -08:00
local code = " return " .. tostring ( self : tree_to_lua ( tree ) . expr ) .. " ; "
2017-09-28 17:49:15 -07:00
if self.debug then
self : writeln ( tostring ( colored.bright ( " RUNNING LUA TO GET VALUE: " ) ) .. " \n " .. tostring ( colored.blue ( colored.bright ( code ) ) ) )
end
2018-01-12 16:33:11 -08:00
local lua_thunk , err = load ( code , nil , nil , self.environment )
2017-09-12 21:38:54 -07:00
if not lua_thunk then
2018-01-12 19:27:59 -08:00
error ( " Failed to compile generated code: \n " .. tostring ( colored.bright ( colored.blue ( colored.onblack ( code ) ) ) ) .. " \n \n " .. tostring ( colored.red ( err ) ) )
2017-09-12 21:38:54 -07:00
end
2018-01-12 16:33:11 -08:00
return lua_thunk ( )
2017-09-12 21:38:54 -07:00
end ,
2018-01-17 16:37:05 -08:00
tree_to_nomsu = function ( self , tree , indentation , max_line , expr_type )
if indentation == nil then
indentation = " "
end
if max_line == nil then
max_line = 80
end
if expr_type == nil then
expr_type = nil
end
assert ( tree , " No tree provided to tree_to_nomsu. " )
assert ( tree.type , " Invalid tree: " .. tostring ( repr ( tree ) ) )
local join_lines
join_lines = function ( lines )
for _index_0 = 1 , # lines do
local line = lines [ _index_0 ]
if # indentation + # line > max_line then
return nil
end
end
return concat ( lines , " \n " .. indentation )
end
local is_operator
is_operator = function ( tok )
2018-01-19 17:28:40 -08:00
return tok and tok.type == " Word " and NOMSU_DEFS.operator : match ( tok.value )
2018-01-17 16:37:05 -08:00
end
local inline_expression , noeol_expression , expression
inline_expression = function ( tok )
local _exp_0 = tok.type
if " Block " == _exp_0 then
if # tok.value > 1 then
return nil
end
local nomsu = inline_expression ( tok.value )
return nomsu and " (: " .. tostring ( nomsu ) .. " ) "
elseif " FunctionCall " == _exp_0 then
local buff = " "
for i , bit in ipairs ( tok.value ) do
if bit.type == " Word " then
if i == 1 or ( is_operator ( bit ) and is_operator ( tok.value [ i - 1 ] ) ) then
buff = buff .. bit.value
else
buff = buff .. ( " " .. bit.value )
end
else
local nomsu = inline_expression ( bit )
if not ( nomsu ) then
return nil
end
if not ( i == 1 or bit.type == " Block " ) then
buff = buff .. " "
end
buff = buff .. ( function ( )
if bit.type == " FunctionCall " then
return " ( " .. nomsu .. " ) "
else
return nomsu
end
end ) ( )
end
end
return buff
elseif " List " == _exp_0 then
local bits = { }
local _list_0 = tok.value
2017-10-22 18:40:49 -07:00
for _index_0 = 1 , # _list_0 do
2018-01-17 16:37:05 -08:00
local bit = _list_0 [ _index_0 ]
local nomsu = inline_expression ( bit )
if not ( nomsu ) then
return nil
end
insert ( bits , nomsu )
2017-10-22 18:40:49 -07:00
end
2018-01-17 16:37:05 -08:00
return " [ " .. concat ( bits , " , " ) .. " ] "
elseif " Dict " == _exp_0 then
local bits = { }
local _list_0 = tok.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
local key_nomsu
if bit.dict_key . type == " Word " then
key_nomsu = bit.dict_key . value
else
key_nomsu = inline_expression ( bit.dict_key )
end
if not ( key_nomsu ) then
return nil
end
if bit.dict_key . type == " FunctionCall " then
key_nomsu = " ( " .. key_nomsu .. " ) "
end
local value_nomsu = inline_expression ( bit.dict_value )
if not ( value_nomsu ) then
return nil
end
insert ( bits , key_nomsu .. " = " .. value_nomsu )
end
return " { " .. concat ( bits , " , " ) .. " } "
elseif " Text " == _exp_0 then
local buff = ' " '
local _list_0 = tok.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
if type ( bit ) == ' string ' then
if bit : find ( " \n " ) then
return nil
end
buff = buff .. bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \\ n " )
else
local nomsu = inline_expression ( bit )
if not ( nomsu ) then
return nil
end
buff = buff .. ( function ( )
2018-01-17 16:45:16 -08:00
if bit.type == " Var " or bit.type == " List " or bit.type == " Dict " then
2018-01-17 16:37:05 -08:00
return " \\ " .. nomsu
else
return " \\ ( " .. nomsu .. " ) "
end
end ) ( )
end
if # buff > max_line then
return nil
end
end
return buff .. ' " '
elseif " Nomsu " == _exp_0 then
local nomsu = inline_expression ( tok.value )
if not nomsu then
return nil
end
return " \\ ( " .. nomsu .. " ) "
elseif " Number " == _exp_0 then
return tostring ( tok.value )
elseif " Var " == _exp_0 then
return " % " .. tok.value
2018-01-15 15:50:18 -08:00
else
2018-01-17 16:37:05 -08:00
return nil
2018-01-15 15:50:18 -08:00
end
2018-01-17 16:37:05 -08:00
end
noeol_expression = function ( tok )
local nomsu = inline_expression ( tok )
if nomsu and # nomsu < max_line then
return nomsu
end
local _exp_0 = tok.type
if " Block " == _exp_0 then
local buff = " : "
local _list_0 = tok.value
for _index_0 = 1 , # _list_0 do
local line = _list_0 [ _index_0 ]
nomsu = expression ( line )
if not ( nomsu ) then
return nil
2017-10-22 18:40:49 -07:00
end
2018-01-17 16:37:05 -08:00
buff = buff .. ( " \n " .. self : indent ( nomsu ) )
end
return buff
elseif " FunctionCall " == _exp_0 then
nomsu = expression ( tok )
if not ( nomsu ) then
return nil
end
return " (..) \n " .. self : indent ( nomsu )
elseif " List " == _exp_0 then
local buff = " [..] "
local line = " \n "
local _list_0 = tok.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
nomsu = inline_expression ( bit )
2018-01-17 16:45:16 -08:00
if line ~= " \n " and # line + # " , " + # nomsu > max_line then
buff = buff .. line
line = " \n "
end
2018-01-17 16:37:05 -08:00
local sep = line == " \n " and " " or " , "
2018-01-17 16:45:16 -08:00
if nomsu then
2018-01-17 16:37:05 -08:00
line = line .. ( sep .. nomsu )
if # line >= max_line then
buff = buff .. line
line = " \n "
end
else
line = line .. ( sep .. expression ( bit ) )
buff = buff .. line
line = " \n "
2017-10-22 18:40:49 -07:00
end
2017-12-04 17:35:47 -08:00
end
2018-01-17 16:37:05 -08:00
if line ~= " \n " then
buff = buff .. line
2017-12-04 17:35:47 -08:00
end
2018-01-17 16:37:05 -08:00
return buff
elseif " Dict " == _exp_0 then
local buff = " {..} "
local line = " \n "
local _list_0 = tok.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
local key_nomsu = inline_expression ( bit.dict_key )
if not ( key_nomsu ) then
return nil
end
if bit.dict_key . type == " FunctionCall " then
key_nomsu = " ( " .. key_nomsu .. " ) "
end
local value_nomsu = inline_expression ( bit.dict_value )
if value_nomsu and # key_nomsu + # value_nomsu < max_line then
line = line .. ( key_nomsu .. " = " .. value_nomsu .. " , " )
if # line >= max_line then
buff = buff .. line
line = " \n "
end
else
line = line .. ( key_nomsu .. " = " .. expression ( bit.dict_value ) )
buff = buff .. line
line = " \n "
end
end
if line ~= " \n " then
buff = buff .. line
2017-10-22 18:40:49 -07:00
end
2018-01-17 16:37:05 -08:00
return buff
elseif " Text " == _exp_0 then
local buff = ' ".." \n '
local _list_0 = tok.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
if type ( bit ) == ' string ' then
buff = buff .. bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \n " )
2017-10-22 18:40:49 -07:00
else
2018-01-17 16:37:05 -08:00
nomsu = inline_expression ( bit )
if not ( nomsu ) then
return nil
end
buff = buff .. ( function ( )
2018-01-17 16:45:16 -08:00
if bit.type == " Var " or bit.type == " List " or bit.type == " Dict " then
2018-01-17 16:37:05 -08:00
return " \\ " .. nomsu
else
return " \\ ( " .. nomsu .. " ) "
end
end ) ( )
2017-10-22 18:40:49 -07:00
end
end
2018-01-17 16:37:05 -08:00
return buff
elseif " Nomsu " == _exp_0 then
nomsu = expression ( tok.value )
if not nomsu then
return nil
end
return " \\ (..) \n " .. self : indent ( nomsu )
elseif " Comment " == _exp_0 then
if tok.value : find ( " \n " ) then
return " #.. " .. tok.value : gsub ( " \n " , " \n " )
2017-10-22 18:40:49 -07:00
else
2018-01-17 16:37:05 -08:00
return " # " .. tok.value
2017-10-22 18:40:49 -07:00
end
else
2018-01-17 16:37:05 -08:00
return inline_expression ( tok )
2017-10-22 18:40:49 -07:00
end
2018-01-17 16:37:05 -08:00
end
expression = function ( tok )
local nomsu = inline_expression ( tok )
if nomsu and # nomsu < max_line then
return nomsu
end
local _exp_0 = tok.type
if " Block " == _exp_0 then
if # tok.value == 1 then
2018-01-17 16:45:16 -08:00
if tok.value [ 1 ] . type == " FunctionCall " then
nomsu = inline_expression ( tok.value [ 1 ] )
else
nomsu = noeol_expression ( tok.value [ 1 ] )
end
2018-01-17 16:37:05 -08:00
if nomsu and # ( nomsu : match ( " [^ \n ]* " ) ) < max_line then
return " : " .. nomsu
2017-10-22 18:40:49 -07:00
end
end
2018-01-17 16:37:05 -08:00
return noeol_expression ( tok )
elseif " FunctionCall " == _exp_0 then
local buff = " "
for i , bit in ipairs ( tok.value ) do
if bit.type == " Word " then
if i == 1 or ( is_operator ( bit ) and is_operator ( tok.value [ i - 1 ] ) ) or buff : sub ( - 2 , - 1 ) == " .. " then
buff = buff .. bit.value
else
buff = buff .. ( " " .. bit.value )
end
else
nomsu = inline_expression ( bit )
if nomsu and # nomsu < max_line then
if bit.type == " FunctionCall " then
nomsu = " ( " .. nomsu .. " ) "
end
else
nomsu = expression ( bit )
if not ( nomsu ) then
return nil
end
if bit.type == " FunctionCall " then
nomsu = " (..) \n " .. self : indent ( nomsu )
end
if i < # tok.value then
nomsu = nomsu .. " \n .. "
end
end
if not ( i == 1 or bit.type == " Block " ) then
buff = buff .. " "
end
buff = buff .. nomsu
end
end
return buff
elseif " File " == _exp_0 then
local lines = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local line = _list_0 [ _index_0 ]
nomsu = expression ( line )
assert ( nomsu , " Failed to produce output for: \n " .. tostring ( colored.yellow ( line.src ) ) )
insert ( lines , nomsu )
end
return concat ( lines , " \n " )
elseif " Comment " == _exp_0 then
if tok.value : find ( " \n " ) then
return " #.. " .. tok.value : gsub ( " \n " , " \n " )
2017-10-22 18:40:49 -07:00
else
2018-01-17 16:37:05 -08:00
return " # " .. tok.value
2017-10-22 18:40:49 -07:00
end
else
2018-01-17 16:37:05 -08:00
return noeol_expression ( tok )
2017-10-22 18:40:49 -07:00
end
end
2018-01-17 16:37:05 -08:00
return expression ( tree )
2017-10-22 18:40:49 -07:00
end ,
2017-12-04 17:35:47 -08:00
value_to_nomsu = function ( self , value )
local _exp_0 = type ( value )
if " nil " == _exp_0 then
return " (nil) "
elseif " bool " == _exp_0 then
return value and " (yes) " or " (no) "
elseif " number " == _exp_0 then
return repr ( value )
elseif " table " == _exp_0 then
2017-12-18 16:25:56 -08:00
if is_list ( value ) then
2017-12-04 17:35:47 -08:00
return " [ " .. tostring ( concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # value do
local v = value [ _index_0 ]
_accum_0 [ _len_0 ] = self : value_to_nomsu ( v )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " , " ) ) .. " ] "
else
2018-01-08 18:53:57 -08:00
return " { " .. tostring ( concat ( ( function ( )
2017-12-04 17:35:47 -08:00
local _accum_0 = { }
local _len_0 = 1
for k , v in pairs ( value ) do
_accum_0 [ _len_0 ] = tostring ( self : value_to_nomsu ( k ) ) .. " = " .. tostring ( self : value_to_nomsu ( v ) )
_len_0 = _len_0 + 1
end
return _accum_0
2018-01-08 18:53:57 -08:00
end ) ( ) , " , " ) ) .. " } "
2017-12-04 17:35:47 -08:00
end
2017-12-14 14:26:24 -08:00
elseif " string " == _exp_0 then
if value == " \n " then
return " ' \\ n' "
elseif not value : find ( [["]] ) and not value : find ( " \n " ) and not value : find ( " \\ " ) then
return " \" " .. value .. " \" "
else
return ' ".." \n ' .. ( self : indent ( value ) )
end
2017-12-04 17:35:47 -08:00
else
return error ( " Unsupported value_to_nomsu type: " .. tostring ( type ( value ) ) )
end
end ,
2018-01-11 03:32:12 -08:00
tree_to_lua = function ( self , tree )
2018-01-12 19:27:59 -08:00
assert ( tree , " No tree provided. " )
2017-09-20 04:21:46 -07:00
if not tree.type then
2018-01-12 19:27:59 -08:00
error ( " Invalid tree: " .. tostring ( repr ( tree ) ) )
2017-09-20 04:21:46 -07:00
end
2017-09-12 21:38:54 -07:00
local _exp_0 = tree.type
if " File " == _exp_0 then
2018-01-08 18:53:57 -08:00
if # tree.value == 1 then
2018-01-11 03:32:12 -08:00
return self : tree_to_lua ( tree.value [ 1 ] )
2018-01-08 18:53:57 -08:00
end
2018-01-23 19:22:20 -08:00
local declared_locals = { }
2017-12-11 17:53:23 -08:00
local lua_bits = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local line = _list_0 [ _index_0 ]
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( line )
2018-01-08 18:53:57 -08:00
if not lua then
2018-01-12 19:27:59 -08:00
error ( " No lua produced by " .. tostring ( repr ( line ) ) )
2018-01-08 18:53:57 -08:00
end
2018-01-23 19:22:20 -08:00
if lua.locals then
local new_locals
do
local _accum_0 = { }
local _len_0 = 1
local _list_1 = lua.locals
for _index_1 = 1 , # _list_1 do
local l = _list_1 [ _index_1 ]
if not declared_locals [ l ] then
_accum_0 [ _len_0 ] = l
_len_0 = _len_0 + 1
end
end
new_locals = _accum_0
end
if # new_locals > 0 then
insert ( lua_bits , " local " .. tostring ( concat ( new_locals , " , " ) ) .. " ; " )
for _index_1 = 1 , # new_locals do
local l = new_locals [ _index_1 ]
declared_locals [ l ] = true
end
end
end
2018-01-08 18:53:57 -08:00
if lua.statements then
insert ( lua_bits , lua.statements )
2018-01-23 19:22:20 -08:00
elseif lua.expr then
2018-01-08 18:53:57 -08:00
insert ( lua_bits , tostring ( lua.expr ) .. " ; " )
2017-12-11 17:53:23 -08:00
end
end
2018-01-08 18:53:57 -08:00
return {
statements = concat ( lua_bits , " \n " )
}
2018-01-15 15:50:18 -08:00
elseif " Comment " == _exp_0 then
return {
statements = " -- " .. tree.value : gsub ( " \n " , " \n -- " )
}
2017-09-24 20:20:43 -07:00
elseif " Nomsu " == _exp_0 then
2018-01-08 18:53:57 -08:00
return {
2018-01-10 20:45:03 -08:00
expr = " nomsu:parse( " .. tostring ( repr ( tree.value . src ) ) .. " , " .. tostring ( repr ( tree : get_line_no ( ) ) ) .. " ).value[1] "
2018-01-08 18:53:57 -08:00
}
elseif " Block " == _exp_0 then
2017-09-28 17:49:15 -07:00
local lua_bits = { }
2018-01-23 19:22:20 -08:00
local locals = { }
2017-09-28 17:49:15 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local arg = _list_0 [ _index_0 ]
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( arg )
2018-01-08 18:53:57 -08:00
if # tree.value == 1 and lua.expr and not lua.statements then
return {
expr = lua.expr
}
2018-01-07 18:45:27 -08:00
end
2018-01-23 19:22:20 -08:00
if lua.locals then
local _list_1 = lua.locals
for _index_1 = 1 , # _list_1 do
local l = _list_1 [ _index_1 ]
locals [ l ] = true
end
end
2018-01-08 18:53:57 -08:00
if lua.statements then
insert ( lua_bits , lua.statements )
2018-01-23 19:22:20 -08:00
elseif lua.expr then
2018-01-08 18:53:57 -08:00
insert ( lua_bits , tostring ( lua.expr ) .. " ; " )
2017-09-28 17:49:15 -07:00
end
end
2018-01-08 18:53:57 -08:00
return {
2018-01-23 19:22:20 -08:00
statements = concat ( lua_bits , " \n " ) ,
locals = ( next ( locals ) and utils.keys ( locals ) or nil )
2018-01-08 18:53:57 -08:00
}
2017-09-12 21:38:54 -07:00
elseif " FunctionCall " == _exp_0 then
2017-12-11 17:53:23 -08:00
insert ( self.compilestack , tree )
2018-01-19 17:28:40 -08:00
local fn = rawget ( self.environment . ACTIONS , tree.stub )
2018-01-12 19:27:59 -08:00
local metadata = self.environment . ACTION_METADATA [ fn ]
if metadata and metadata.compile_time then
2018-01-11 02:07:37 -08:00
local args
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local arg = _list_0 [ _index_0 ]
if arg.type ~= " Word " then
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
2018-01-18 01:49:13 -08:00
if metadata and metadata.arg_orders then
2018-01-12 19:27:59 -08:00
local new_args
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = metadata.arg_orders [ tree.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
new_args = _accum_0
end
args = new_args
end
2018-01-11 02:07:37 -08:00
if self.debug then
self : write ( tostring ( colored.bright ( " RUNNING MACRO " ) ) .. " " .. tostring ( colored.underscore ( colored.magenta ( tree.stub ) ) ) .. " " )
2018-01-11 18:48:43 -08:00
self : writeln ( tostring ( colored.bright ( " WITH ARGS: " ) ) .. " " .. tostring ( colored.dim ( repr ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # args do
local a = args [ _index_0 ]
_accum_0 [ _len_0 ] = ( repr ( a ) ) : sub ( 1 , 50 )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) ) )
2018-01-11 02:07:37 -08:00
end
2018-01-12 19:27:59 -08:00
local lua = fn ( unpack ( args ) )
2017-12-11 17:53:23 -08:00
remove ( self.compilestack )
2018-01-08 18:53:57 -08:00
return lua
2018-01-12 19:27:59 -08:00
elseif not metadata and self.__class . math_patt : match ( tree.stub ) then
2018-01-07 18:03:37 -08:00
local bits = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local tok = _list_0 [ _index_0 ]
if tok.type == " Word " then
insert ( bits , tok.value )
else
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( tok )
2018-01-23 19:22:20 -08:00
assert ( lua.expr , " non-expression value inside math expression: " .. tostring ( tok.src ) )
2018-01-08 18:53:57 -08:00
insert ( bits , lua.expr )
2018-01-07 18:03:37 -08:00
end
end
2018-01-08 18:53:57 -08:00
remove ( self.compilestack )
return {
expr = " ( " .. tostring ( concat ( bits , " " ) ) .. " ) "
}
2017-09-24 20:20:43 -07:00
end
2018-01-11 01:03:52 -08:00
local args = { }
2017-09-24 20:20:43 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local _continue_0 = false
repeat
2018-01-11 01:03:52 -08:00
local tok = _list_0 [ _index_0 ]
if tok.type == " Word " then
2017-09-24 20:20:43 -07:00
_continue_0 = true
break
end
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( tok )
2018-01-23 19:22:20 -08:00
assert ( lua.expr , tostring ( tree : get_line_no ( ) ) .. " : Cannot use: \n " .. tostring ( colored.yellow ( tok.src ) ) .. " \n as an argument to " .. tostring ( tree.stub ) .. " , since it's not an expression, it produces: " .. tostring ( repr ( lua ) ) )
2018-01-11 01:03:52 -08:00
insert ( args , lua.expr )
2017-09-24 20:20:43 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
2017-09-12 21:38:54 -07:00
end
end
2018-01-18 01:49:13 -08:00
if metadata and metadata.arg_orders then
2018-01-11 01:03:52 -08:00
local new_args
do
local _accum_0 = { }
local _len_0 = 1
2018-01-12 19:27:59 -08:00
local _list_1 = metadata.arg_orders [ tree.stub ]
2018-01-11 01:03:52 -08:00
for _index_0 = 1 , # _list_1 do
local p = _list_1 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p ]
_len_0 = _len_0 + 1
end
new_args = _accum_0
end
args = new_args
end
2017-12-11 17:53:23 -08:00
remove ( self.compilestack )
2018-01-08 18:53:57 -08:00
return {
2018-01-12 19:27:59 -08:00
expr = self.__class : comma_separated_items ( " ACTIONS[ " .. tostring ( repr ( tree.stub ) ) .. " ]( " , args , " ) " )
2018-01-08 18:53:57 -08:00
}
2018-01-11 01:09:26 -08:00
elseif " Text " == _exp_0 then
2017-09-14 02:41:10 -07:00
local concat_parts = { }
local string_buffer = " "
2017-09-24 20:20:43 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local _continue_0 = false
repeat
local bit = _list_0 [ _index_0 ]
2017-09-14 02:41:10 -07:00
if type ( bit ) == " string " then
2017-09-24 20:20:43 -07:00
string_buffer = string_buffer .. bit
_continue_0 = true
break
end
if string_buffer ~= " " then
insert ( concat_parts , repr ( string_buffer ) )
string_buffer = " "
2017-09-14 02:41:10 -07:00
end
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( bit )
2017-09-28 17:49:15 -07:00
if self.debug then
self : writeln ( ( colored.bright ( " INTERP: " ) ) )
self : print_tree ( bit )
2018-01-08 18:53:57 -08:00
self : writeln ( tostring ( colored.bright ( " EXPR: " ) ) .. " " .. tostring ( lua.expr ) .. " , " .. tostring ( colored.bright ( " STATEMENT: " ) ) .. " " .. tostring ( lua.statements ) )
2017-09-28 17:49:15 -07:00
end
2018-01-23 19:22:20 -08:00
assert ( lua.expr , " Cannot use [[ " .. tostring ( bit.src ) .. " ]] as a string interpolation value, since it's not an expression. " )
2018-01-12 16:33:42 -08:00
insert ( concat_parts , " stringify( " .. tostring ( lua.expr ) .. " ) " )
2017-09-24 20:20:43 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
2017-09-14 02:41:10 -07:00
end
end
if string_buffer ~= " " then
2017-09-21 21:11:13 -07:00
insert ( concat_parts , repr ( string_buffer ) )
2017-09-14 02:41:10 -07:00
end
2017-09-26 15:27:01 -07:00
if # concat_parts == 0 then
2018-01-08 18:53:57 -08:00
return {
expr = " '' "
}
2017-09-28 17:49:15 -07:00
elseif # concat_parts == 1 then
2018-01-08 18:53:57 -08:00
return {
expr = concat_parts [ 1 ]
}
2017-09-28 17:49:15 -07:00
else
2018-01-08 18:53:57 -08:00
return {
expr = " ( " .. tostring ( concat ( concat_parts , " .. " ) ) .. " ) "
}
2017-09-26 15:27:01 -07:00
end
2017-09-12 21:38:54 -07:00
elseif " List " == _exp_0 then
2017-09-24 20:20:43 -07:00
local items = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local item = _list_0 [ _index_0 ]
2018-01-11 03:32:12 -08:00
local lua = self : tree_to_lua ( item )
2018-01-23 19:22:20 -08:00
assert ( lua.expr , " Cannot use [[ " .. tostring ( item.src ) .. " ]] as a list item, since it's not an expression. " )
2018-01-08 18:53:57 -08:00
insert ( items , lua.expr )
2017-09-12 21:38:54 -07:00
end
2018-01-08 18:53:57 -08:00
return {
expr = self.__class : comma_separated_items ( " { " , items , " } " )
}
2018-01-03 00:52:01 -08:00
elseif " Dict " == _exp_0 then
local items = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local entry = _list_0 [ _index_0 ]
2018-01-08 18:53:57 -08:00
local key_lua
2018-01-03 00:52:01 -08:00
if entry.dict_key . type == " Word " then
2018-01-08 18:53:57 -08:00
key_lua = {
expr = repr ( entry.dict_key . value )
}
2018-01-03 00:52:01 -08:00
else
2018-01-11 03:32:12 -08:00
key_lua = self : tree_to_lua ( entry.dict_key )
2018-01-03 00:52:01 -08:00
end
2018-01-23 19:22:20 -08:00
assert ( key_lua.expr , " Cannot use [[ " .. tostring ( entry.dict_key . src ) .. " ]] as a dict key, since it's not an expression. " )
2018-01-11 03:32:12 -08:00
local value_lua = self : tree_to_lua ( entry.dict_value )
2018-01-23 19:22:20 -08:00
assert ( value_lua.expr , " Cannot use [[ " .. tostring ( entry.dict_value . src ) .. " ]] as a dict value, since it's not an expression. " )
2018-01-08 18:53:57 -08:00
local key_str = key_lua.expr : match ( [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=] )
2018-01-03 00:52:01 -08:00
if key_str then
2018-01-08 18:53:57 -08:00
insert ( items , tostring ( key_str ) .. " = " .. tostring ( value_lua.expr ) )
2018-01-18 16:44:11 -08:00
elseif key_lua.expr : sub ( 1 , 1 ) == " [ " then
insert ( items , " [ " .. tostring ( key_lua.expr ) .. " ]= " .. tostring ( value_lua.expr ) )
2018-01-03 00:52:01 -08:00
else
2018-01-08 18:53:57 -08:00
insert ( items , " [ " .. tostring ( key_lua.expr ) .. " ]= " .. tostring ( value_lua.expr ) )
2018-01-03 00:52:01 -08:00
end
end
2018-01-08 18:53:57 -08:00
return {
expr = self.__class : comma_separated_items ( " { " , items , " } " )
}
2017-09-24 20:20:43 -07:00
elseif " Number " == _exp_0 then
2018-01-08 18:53:57 -08:00
return {
expr = repr ( tree.value )
}
2017-09-12 21:38:54 -07:00
elseif " Var " == _exp_0 then
2018-01-10 20:45:03 -08:00
return {
2018-01-12 19:27:59 -08:00
expr = self : var_to_lua_identifier ( tree.value )
2018-01-10 20:45:03 -08:00
}
2017-09-12 21:38:54 -07:00
else
2018-01-12 19:27:59 -08:00
return error ( " Unknown/unimplemented thingy: " .. tostring ( tree.type ) )
2017-09-12 21:38:54 -07:00
end
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 )
2017-09-24 20:20:43 -07:00
if type ( tree ) ~= ' table ' or not tree.type then
return
end
local _exp_0 = tree.type
2018-01-11 01:09:26 -08:00
if " List " == _exp_0 or " File " == _exp_0 or " Block " == _exp_0 or " FunctionCall " == _exp_0 or " Text " == _exp_0 then
2017-09-24 20:20:43 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
2017-09-26 15:27:01 -07:00
self : walk_tree ( v , depth + 1 )
2017-09-24 20:20:43 -07:00
end
2018-01-03 00:52:01 -08:00
elseif " Dict " == _exp_0 then
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local e = _list_0 [ _index_0 ]
self : walk_tree ( e.dict_key , depth + 1 )
self : walk_tree ( e.dict_value , depth + 1 )
end
2017-09-24 20:20:43 -07:00
else
2017-09-26 15:27:01 -07:00
self : walk_tree ( tree.value , depth + 1 )
end
return nil
end ,
print_tree = function ( self , tree )
2017-09-28 17:49:15 -07:00
self : write ( colors.bright .. colors.green )
2017-09-26 15:27:01 -07:00
for node , depth in coroutine.wrap ( function ( )
return self : walk_tree ( tree )
end ) do
if type ( node ) ~= ' table ' or not node.type then
self : writeln ( ( " " ) : rep ( depth ) .. repr ( node ) )
else
self : writeln ( tostring ( ( " " ) : rep ( depth ) ) .. tostring ( node.type ) .. " : " )
end
end
2017-09-28 17:49:15 -07:00
return self : write ( colors.reset )
2017-09-26 15:27:01 -07:00
end ,
tree_to_str = function ( self , tree )
local bits = { }
for node , depth in coroutine.wrap ( function ( )
return self : walk_tree ( tree )
end ) do
if type ( node ) ~= ' table ' or not node.type then
insert ( bits , ( ( " " ) : rep ( depth ) .. repr ( node ) ) )
else
insert ( bits , ( tostring ( ( " " ) : rep ( depth ) ) .. tostring ( node.type ) .. " : " ) )
end
2017-09-24 20:20:43 -07:00
end
2017-09-26 15:27:01 -07:00
return concat ( bits , " \n " )
2017-09-24 20:20:43 -07:00
end ,
2018-01-12 16:45:36 -08:00
tree_with_replaced_vars = function ( self , tree , replacements )
2017-09-24 20:20:43 -07:00
if type ( tree ) ~= ' table ' then
return tree
end
local _exp_0 = tree.type
if " Var " == _exp_0 then
2018-01-12 16:45:36 -08:00
if replacements [ tree.value ] ~= nil then
tree = replacements [ tree.value ]
2017-09-24 20:20:43 -07:00
end
2018-01-11 01:09:26 -08:00
elseif " File " == _exp_0 or " Nomsu " == _exp_0 or " Block " == _exp_0 or " List " == _exp_0 or " FunctionCall " == _exp_0 or " Text " == _exp_0 then
2018-01-12 16:45:36 -08:00
local new_value = self : tree_with_replaced_vars ( tree.value , replacements )
2017-09-24 20:20:43 -07:00
if new_value ~= tree.value then
do
local _tbl_0 = { }
for k , v in pairs ( tree ) do
_tbl_0 [ k ] = v
end
tree = _tbl_0
end
tree.value = new_value
end
2018-01-03 00:52:01 -08:00
elseif " Dict " == _exp_0 then
local dirty = false
2018-01-12 16:45:36 -08:00
replacements = { }
2018-01-03 00:52:01 -08:00
for i , e in ipairs ( tree.value ) do
2018-01-12 16:45:36 -08:00
local new_key = self : tree_with_replaced_vars ( e.dict_key , replacements )
local new_value = self : tree_with_replaced_vars ( e.dict_value , replacements )
2018-01-03 00:52:01 -08:00
dirty = dirty or ( new_key ~= e.dict_key or new_value ~= e.dict_value )
replacements [ i ] = {
dict_key = new_key ,
dict_value = new_value
}
end
if dirty then
do
local _tbl_0 = { }
for k , v in pairs ( tree ) do
_tbl_0 [ k ] = v
end
tree = _tbl_0
end
tree.value = replacements
end
2017-09-24 20:20:43 -07:00
elseif nil == _exp_0 then
local new_values = { }
local any_different = false
for k , v in pairs ( tree ) do
2018-01-12 16:45:36 -08:00
new_values [ k ] = self : tree_with_replaced_vars ( v , replacements )
2017-09-24 20:20:43 -07:00
any_different = any_different or ( new_values [ k ] ~= tree [ k ] )
end
if any_different then
tree = new_values
end
end
return tree
end ,
2017-09-26 15:27:01 -07:00
get_stub = function ( self , x )
2017-09-21 21:11:13 -07:00
if not x then
2018-01-12 19:27:59 -08:00
error ( " Nothing to get stub from " )
2017-09-21 21:11:13 -07:00
end
if type ( x ) == ' string ' then
2018-01-05 14:56:35 -08:00
local spec = concat ( self.__class . stub_patt : match ( x ) , " " )
2018-01-12 19:27:59 -08:00
local arg_names = { }
2018-01-18 01:49:13 -08:00
local stub = spec : gsub ( " %%(%S*) " , function ( arg )
2018-01-12 19:27:59 -08:00
insert ( arg_names , arg )
return " % "
end )
return stub , arg_names
2017-12-04 17:35:47 -08:00
end
if type ( x ) ~= ' table ' then
2018-01-12 19:27:59 -08:00
error ( " Invalid type for getting stub: " .. tostring ( type ( x ) ) .. " for: \n " .. tostring ( repr ( x ) ) )
2017-09-21 21:11:13 -07:00
end
local _exp_0 = x.type
2018-01-11 01:09:26 -08:00
if " Text " == _exp_0 then
2017-09-26 15:27:01 -07:00
return self : get_stub ( x.value )
2017-09-21 21:11:13 -07:00
elseif " FunctionCall " == _exp_0 then
2017-12-04 17:35:47 -08:00
return self : get_stub ( x.src )
2017-10-02 19:00:58 -07:00
else
2018-01-12 19:27:59 -08:00
return error ( " Unsupported get stub type: " .. tostring ( x.type ) .. " for " .. tostring ( repr ( x ) ) )
2017-09-12 21:38:54 -07:00
end
end ,
2017-10-02 19:00:58 -07:00
get_stubs = function ( self , x )
if type ( x ) ~= ' table ' then
return {
{
self : get_stub ( x )
}
}
end
local _exp_0 = x.type
if nil == _exp_0 then
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # x do
local i = x [ _index_0 ]
_accum_0 [ _len_0 ] = {
self : get_stub ( i )
}
_len_0 = _len_0 + 1
end
return _accum_0
elseif " List " == _exp_0 then
local _accum_0 = { }
local _len_0 = 1
local _list_0 = x.value
for _index_0 = 1 , # _list_0 do
local i = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = {
self : get_stub ( i )
}
_len_0 = _len_0 + 1
end
return _accum_0
end
return {
{
self : get_stub ( x )
}
}
end ,
2017-09-21 00:10:26 -07:00
var_to_lua_identifier = function ( self , var )
2017-09-24 20:20:43 -07:00
if type ( var ) == ' table ' and var.type == " Var " then
var = var.value
2017-09-21 00:10:26 -07:00
end
2018-01-12 19:27:59 -08:00
return " _ " .. ( var : gsub ( " %W " , function ( verboten )
2017-09-21 04:04:08 -07:00
if verboten == " _ " then
return " __ "
else
return ( " _%x " ) : format ( verboten : byte ( ) )
end
end ) )
2017-09-21 00:10:26 -07:00
end ,
2017-12-11 17:53:23 -08:00
source_code = function ( self , level )
if level == nil then
level = 0
end
return self : dedent ( self.compilestack [ # self.compilestack - level ] . src )
end ,
2017-09-12 21:38:54 -07:00
initialize_core = function ( self )
2018-01-12 16:33:11 -08:00
local nomsu = self
2017-10-19 18:16:48 -07:00
local nomsu_string_as_lua
2018-01-12 16:33:11 -08:00
nomsu_string_as_lua = function ( code )
2017-10-19 18:16:48 -07:00
local concat_parts = { }
local _list_0 = code.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
if type ( bit ) == " string " then
insert ( concat_parts , bit )
else
2018-01-12 16:33:11 -08:00
local lua = nomsu : tree_to_lua ( bit )
2018-01-23 19:22:20 -08:00
assert ( lua.expr , " Cannot use [[ " .. tostring ( bit.src ) .. " ]] as a string interpolation value, since it's not an expression. " )
2018-01-08 18:53:57 -08:00
insert ( concat_parts , lua.expr )
2017-10-19 18:16:48 -07:00
end
end
return concat ( concat_parts )
end
2018-01-12 16:33:11 -08:00
self : define_compile_action ( " immediately %block " , " nomsu.moon " , function ( _block )
local lua = nomsu : tree_to_lua ( _block )
2018-01-08 18:53:57 -08:00
local lua_code = lua.statements or ( lua.expr .. " ; " )
2018-01-09 14:59:06 -08:00
lua_code = " -- Immediately: \n " .. lua_code
2018-01-12 16:33:11 -08:00
nomsu : run_lua ( lua_code )
2018-01-08 18:53:57 -08:00
return {
2018-01-23 19:22:20 -08:00
statements = lua_code ,
locals = lua.locals
2018-01-08 18:53:57 -08:00
}
end )
2018-01-12 16:33:11 -08:00
self : define_compile_action ( " lua> %code " , " nomsu.moon " , function ( _code )
local lua = nomsu_string_as_lua ( _code )
2018-01-08 18:53:57 -08:00
return {
statements = lua
}
end )
2018-01-12 16:33:11 -08:00
self : define_compile_action ( " =lua %code " , " nomsu.moon " , function ( _code )
local lua = nomsu_string_as_lua ( _code )
2018-01-08 18:53:57 -08:00
return {
expr = lua
}
2017-12-09 16:00:09 -08:00
end )
2018-01-12 16:33:11 -08:00
self : define_compile_action ( " __line_no__ " , " nomsu.moon " , function ( )
2018-01-11 01:03:52 -08:00
return {
2018-01-12 16:33:11 -08:00
expr = repr ( nomsu.compilestack [ # nomsu.compilestack ] : get_line_no ( ) )
2018-01-11 01:03:52 -08:00
}
end )
2018-01-12 16:33:11 -08:00
self : define_compile_action ( " __src__ %level " , " nomsu.moon " , function ( _level )
2018-01-08 18:53:57 -08:00
return {
2018-01-12 16:33:11 -08:00
expr = repr ( nomsu : source_code ( nomsu : tree_to_value ( _level ) ) )
2018-01-08 18:53:57 -08:00
}
2017-12-30 14:31:07 -08:00
end )
2018-01-12 16:33:11 -08:00
self : define_action ( " run file %filename " , " nomsu.moon " , function ( _filename )
2018-01-23 15:25:27 -08:00
return nomsu : run_file ( _filename )
2018-01-10 16:22:45 -08:00
end )
2018-01-12 16:33:11 -08:00
return self : define_compile_action ( " use %filename " , " nomsu.moon " , function ( _filename )
local filename = nomsu : tree_to_value ( _filename )
nomsu : require_file ( filename )
2018-01-08 18:53:57 -08:00
return {
2018-01-10 16:22:45 -08:00
statements = " nomsu:require_file( " .. tostring ( repr ( filename ) ) .. " ); "
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 )
2017-09-20 04:21:46 -07:00
self.write = function ( self , ... )
return io.write ( ... )
end
2017-10-12 14:39:49 -07:00
self.write_err = function ( self , ... )
return io.stderr : write ( ... )
end
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
} )
2017-12-11 17:53:23 -08:00
self.compilestack = { }
2017-09-12 21:38:54 -07:00
self.debug = false
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-01-19 17:28:40 -08:00
self.environment . ACTIONS = setmetatable ( { } , {
__index = function ( self , key )
return error ( " Attempt to run undefined action: " .. tostring ( key ) , 0 )
end
} )
self.action_metadata = setmetatable ( { } , {
__mode = " k "
} )
self.environment . ACTION_METADATA = self.action_metadata
self.environment . LOADED = { }
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
2017-12-04 17:35:47 -08:00
self.def_number = 0
2018-01-07 18:03:37 -08:00
self.math_patt = re.compile ( [[ "%" (" " [*/^+-] " %")+ ]] )
2017-09-21 13:30:59 -07:00
self.unescape_string = function ( self , str )
2018-01-19 17:28:40 -08:00
return Cs ( ( ( P ( " \\ \\ " ) / " \\ " ) + ( P ( " \\ \" " ) / ' " ' ) + NOMSU_DEFS.escaped_char + P ( 1 ) ) ^ 0 ) : match ( str )
2017-09-21 13:30:59 -07:00
end
2017-09-12 21:38:54 -07:00
self.comma_separated_items = function ( self , open , items , close )
2017-09-21 13:30:59 -07:00
local bits = {
open
}
local so_far = 0
for i , item in ipairs ( items ) do
if i < # items then
item = item .. " , "
end
insert ( bits , item )
so_far = so_far + # item
if so_far >= 80 then
insert ( bits , " \n " )
so_far = 0
2017-09-12 21:38:54 -07:00
end
2017-09-21 13:30:59 -07:00
end
insert ( bits , close )
2017-09-21 21:11:13 -07:00
return concat ( bits )
2017-09-12 21:38:54 -07:00
end
2018-01-12 19:27:59 -08:00
self.stub_patt = re.compile ( " {|(' '+ / ' \n ..' / {'%' %id*} / {%id+} / {%op})*|} " , {
2018-01-19 17:28:40 -08:00
id = NOMSU_DEFS.ident_char ,
op = NOMSU_DEFS.operator
2018-01-05 14:56:35 -08:00
} )
2017-09-13 16:22:04 -07:00
NomsuCompiler = _class_0
2017-09-12 21:38:54 -07:00
end
2017-10-08 18:23:48 -07:00
if arg then
2017-10-08 20:41:05 -07:00
colors = require ( ' consolecolors ' )
2017-10-08 18:25:50 -07:00
local parser = re.compile ( [ [ args <- { | { : flags : flags ? : } ( { : input : input : } " ; " ( " -o; " { : output : output : } " ; " ) ? ) ? ( " ; " ) ? | } ! .
2017-10-08 18:23:48 -07:00
flags <- ( ( { | ( { flag } " ; " ) * | } ) -> set )
2018-01-07 18:45:27 -08:00
flag <- " -c " / " -i " / " -p " / " -O " / " --help " / " -h " / " -v "
2017-10-08 18:23:48 -07:00
input <- " - " / [ ^ ; ] +
output <- " - " / [ ^ ; ] +
] ] , {
2017-12-18 16:25:56 -08:00
set = set
2017-10-08 18:23:48 -07:00
} )
local args = concat ( arg , " ; " ) .. " ; "
args = parser : match ( args ) or { }
if not args or not args.flags or args.flags [ " --help " ] or args.flags [ " -h " ] then
2017-12-14 14:08:07 -08:00
print ( " Usage: lua nomsu.lua [-c] [-i] [-p] [-O] [--help] [input [-o output]] " )
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 ( )
local run
run = function ( )
if args.flags [ " -v " ] then
nomsu.debug = true
2017-10-08 18:23:48 -07:00
end
2018-01-11 01:03:52 -08:00
nomsu.skip_precompiled = not args.flags [ " -O " ]
if args.input then
if args.flags [ " -c " ] and not args.output then
args.output = args.input : gsub ( " %.nom " , " .lua " )
end
local compiled_output = nil
if args.flags [ " -p " ] then
local _write = nomsu.write
nomsu.write = function ( ) end
compiled_output = io.output ( )
elseif args.output then
compiled_output = io.open ( args.output , ' w ' )
end
if args.input : match ( " .*%.lua " ) then
local retval = dofile ( args.input ) ( nomsu , { } )
2017-10-08 18:23:48 -07:00
else
2018-01-11 01:03:52 -08:00
local input
if args.input == ' - ' then
input = io.read ( ' *a ' )
else
input = io.open ( args.input ) : read ( " *a " )
end
local retval , code = nomsu : run ( input , args.input )
if args.output then
compiled_output : write ( code )
end
2017-10-08 18:23:48 -07:00
end
2018-01-11 01:03:52 -08:00
if args.flags [ " -p " ] then
nomsu.write = _write
2018-01-08 18:53:57 -08:00
end
2017-10-08 18:23:48 -07:00
end
2018-01-11 01:03:52 -08:00
if args.flags [ " -i " ] then
2018-01-11 14:07:14 -08:00
nomsu : run ( ' use "lib/core.nom" ' , " stdin " )
2018-01-11 01:03:52 -08:00
while true do
local buff = " "
while true do
io.write ( " >> " )
local line = io.read ( " *L " )
if line == " \n " or not line then
break
end
buff = buff .. line
end
if # buff == 0 then
break
end
local ok , ret = pcall ( function ( )
return nomsu : run ( buff , " stdin " )
end )
if ok and ret ~= nil then
print ( " = " .. repr ( ret ) )
end
end
2017-09-12 21:38:54 -07:00
end
end
2018-01-11 01:03:52 -08:00
local err_hand
err_hand = function ( error_message )
print ( tostring ( colored.red ( " ERROR: " ) ) .. " " .. tostring ( colored.bright ( colored.yellow ( colored.onred ( ( error_message or " " ) ) ) ) ) )
print ( " stack traceback: " )
local to_lua
to_lua = require ( " moonscript.base " ) . to_lua
local nomsu_file = io.open ( " nomsu.moon " )
local nomsu_source = nomsu_file : read ( " *a " )
local _ , line_table = to_lua ( nomsu_source )
nomsu_file : close ( )
local level = 2
2017-09-14 15:35:06 -07:00
while true do
2018-01-11 01:03:52 -08:00
local _continue_0 = false
repeat
local calling_fn = debug.getinfo ( level )
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
do
2018-01-12 19:27:59 -08:00
local metadata = nomsu.action_metadata [ calling_fn.func ]
if metadata then
line = colored.yellow ( metadata.line_no )
name = colored.bright ( colored.yellow ( metadata.aliases [ 1 ] ) )
2018-01-11 01:03:52 -08:00
else
if calling_fn.istailcall and not name then
name = " <tail call> "
end
if calling_fn.short_src == " ./nomsu.moon " then
2018-01-19 17:28:40 -08:00
local char = line_table [ calling_fn.currentline ]
local line_num = 1
2018-01-11 01:03:52 -08:00
for _ in nomsu_source : sub ( 1 , char ) : gmatch ( " \n " ) do
line_num = line_num + 1
end
line = colored.cyan ( tostring ( calling_fn.short_src ) .. " : " .. tostring ( line_num ) )
name = colored.bright ( colored.cyan ( name or " ??? " ) )
else
2018-01-19 17:28:40 -08:00
line = colored.blue ( tostring ( calling_fn.short_src ) .. " : " .. tostring ( calling_fn.currentline ) )
2018-01-11 01:03:52 -08:00
name = colored.bright ( colored.blue ( name or " ??? " ) )
end
end
end
local _from = colored.dim ( colored.white ( " | " ) )
print ( ( " %32s %s %s " ) : format ( name , _from , line ) )
_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-01-11 01:03:52 -08:00
return os.exit ( false , true )
2017-09-14 15:35:06 -07:00
end
2018-01-11 01:03:52 -08:00
xpcall ( run , err_hand )
2017-09-12 21:38:54 -07:00
end
2017-09-13 16:22:04 -07:00
return NomsuCompiler