2018-02-02 13:59:58 -08:00
local lfs = require ( ' lfs ' )
2017-09-12 21:38:54 -07:00
local re = require ( ' re ' )
local lpeg = require ( ' lpeg ' )
2018-04-11 20:05:12 -07:00
lpeg.setmaxstack ( 10000 )
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
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-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-04-20 16:23:53 -07:00
local contents = file : read ( " a " ) : sub ( 1 , - 2 )
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
} )
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 )
if type ( i ) == ' number ' then
return string.sub ( self , i , i )
elseif type ( i ) == ' table ' then
return string.sub ( self , i [ 1 ] , i [ 2 ] )
else
return string [ i ]
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-03-05 18:44:26 -08:00
_with_0.Tuple = function ( values )
return Tuple ( table.unpack ( values ) )
end
2018-02-13 15:17:45 -08:00
_with_0.DictEntry = function ( k , v )
return Types.DictEntry ( k , v )
end
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 ]
local spaces = self : match ( " [ ]* " , start )
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-01-19 17:28:40 -08:00
_with_0.error = function ( src , pos , err_msg )
2018-04-18 15:28:46 -07:00
if src : sub ( pos , pos ) : match ( " [ \r \n ] " ) then
pos = pos + # src : match ( " [ \t \n \r ]* " , pos )
2018-01-19 17:28:40 -08:00
end
local line_no = 1
2018-04-18 15:28:46 -07:00
local text_loc = lpeg.userdata . source_code.source : sub ( pos , pos )
2018-04-11 20:05:12 -07:00
line_no = text_loc : get_line_number ( )
2018-04-19 17:23:44 -07:00
src = FILE_CACHE [ text_loc.filename ]
2018-04-11 20:05:12 -07:00
local prev_line = src : sub ( LINE_STARTS [ src ] [ line_no - 1 ] or 1 , LINE_STARTS [ src ] [ line_no ] - 1 )
local err_line = src : sub ( LINE_STARTS [ src ] [ line_no ] , ( LINE_STARTS [ src ] [ line_no + 1 ] or 0 ) - 1 )
local next_line = src : sub ( LINE_STARTS [ src ] [ line_no + 1 ] or - 1 , ( LINE_STARTS [ src ] [ line_no + 2 ] or 0 ) - 1 )
2018-04-11 21:07:13 -07:00
local pointer = ( " - " ) : rep ( pos - LINE_STARTS [ src ] [ line_no ] ) .. " ^ "
2018-04-18 15:28:46 -07:00
err_msg = ( err_msg or " Parse error " ) .. " in " .. tostring ( lpeg.userdata . source_code.source . filename ) .. " on line " .. tostring ( line_no ) .. " : \n "
2018-01-19 17:28:40 -08:00
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
NOMSU_DEFS = _with_0
end
setmetatable ( NOMSU_DEFS , {
__index = function ( self , key )
local make_node
make_node = function ( start , value , stop )
2018-02-13 15:17:45 -08:00
if type ( value ) == ' table ' then
2018-04-06 16:45:51 -07:00
error ( " Not a tuple: " .. tostring ( repr ( value ) ) )
2018-02-13 15:17:45 -08:00
end
2018-04-20 16:23:53 -07:00
local source = lpeg.userdata . source_code.source
start = start + ( source.start - 1 )
stop = stop + ( source.start - 1 )
source = Source ( source.filename , start , stop - 1 )
2018-04-18 15:28:46 -07:00
local node = Types [ key ] ( value , source )
2018-02-08 16:22:57 -08:00
return node
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 } " ) " ( " " * ) " : "
{ ( ( % nl " " + [ ^% nl ] * ) + ) / ( [ ^% nl ] * ) } ) -> " %1 <- ({} %3 {}) -> %2 "
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-04-18 15:45:58 -07:00
local _nomsu_chunk_counter , stub_defs , stub_pattern , var_pattern
2017-09-12 21:38:54 -07:00
local _base_0 = {
2018-01-25 17:34:49 -08:00
define_action = function ( self , signature , source , fn )
if type ( fn ) ~= ' function ' then
error ( ' function ' , " Bad fn: " .. tostring ( repr ( fn ) ) )
2017-09-12 21:38:54 -07:00
end
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
}
elseif type ( signature ) ~= ' table ' or signature.type ~= nil then
error ( " Invalid signature, expected list of strings, but got: " .. tostring ( repr ( signature ) ) , 0 )
2017-10-13 16:10:47 -07:00
end
2018-01-25 17:34:49 -08:00
local stubs = self : get_stubs_from_signature ( signature )
local stub_args = self : get_args_from_signature ( signature )
2018-04-08 18:11:44 -07: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
2018-01-25 17:34:49 -08:00
for sig_i = 1 , # stubs do
local stub , args = stubs [ sig_i ] , stub_args [ sig_i ]
2018-02-06 22:06:39 -08:00
self.environment . ACTIONS [ stub ] = fn
2018-01-18 01:49:13 -08:00
if not ( fn_info.isvararg ) then
local arg_positions
do
local _accum_0 = { }
local _len_0 = 1
2018-01-25 17:34:49 -08:00
for _index_0 = 1 , # args do
local a = args [ _index_0 ]
_accum_0 [ _len_0 ] = fn_arg_positions [ a ]
2018-01-18 01:49:13 -08:00
_len_0 = _len_0 + 1
end
arg_positions = _accum_0
end
arg_orders [ stub ] = arg_positions
2017-12-04 17:35:47 -08:00
end
2017-09-18 22:41:50 -07:00
end
2018-02-08 16:22:57 -08:00
self.action_metadata [ fn ] = {
2018-01-12 19:27:59 -08:00
fn = fn ,
2018-01-25 17:34:49 -08:00
source = source ,
aliases = stubs ,
2018-01-12 19:27:59 -08:00
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-25 17:34:49 -08:00
define_compile_action = function ( self , signature , source , fn , src )
self : define_action ( signature , source , fn )
2018-02-08 16:22:57 -08:00
self.action_metadata [ fn ] . compile_time = true
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-25 17:34:49 -08:00
return error ( " Not currently functional. " , 0 )
2017-10-31 16:19:08 -07:00
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
2018-01-25 17:34:49 -08:00
if line : match ( " ^%s*#.* " ) or line : match ( " ^%s*$ " ) then
2017-12-04 17:35:47 -08:00
_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 " ) )
2018-01-25 17:34:49 -08:00
elseif indent_spaces ~= math.huge then
2017-12-04 17:35:47 -08:00
return ( code : gsub ( " \n " .. ( " " ) : rep ( indent_spaces ) , " \n " ) )
2018-01-25 17:34:49 -08:00
else
return code
2017-12-04 17:35:47 -08:00
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-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-04-11 21:07:13 -07:00
}
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 ) ) ) )
2017-09-12 21:38:54 -07:00
return tree
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-04-18 15:45:58 -07:00
if # 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-01-12 19:27:59 -08:00
assert ( tree.type == " File " , " Attempt to run non-file: " .. tostring ( tree.type ) )
2018-04-25 15:37:13 -07:00
local lua = tree : as_lua ( self )
2018-04-17 14:18:23 -07:00
lua : convert_to_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-19 19:43:23 -07:00
run_file = function ( self , filename , compile_fn )
if compile_fn == nil then
compile_fn = nil
end
2018-02-02 15:48:28 -08:00
local file_attributes = assert ( lfs.attributes ( filename ) , " File not found: " .. tostring ( filename ) )
if file_attributes.mode == " directory " then
for short_filename in lfs.dir ( filename ) do
local full_filename = filename .. ' / ' .. short_filename
local attr = lfs.attributes ( full_filename )
if attr.mode ~= " directory " and short_filename : match ( " .*%.nom " ) then
2018-04-19 19:43:23 -07:00
self : run_file ( full_filename , compile_fn )
2018-02-02 15:48:28 -08:00
end
end
return
end
2018-01-10 16:22:45 -08:00
if filename : match ( " .*%.lua " ) then
2018-04-11 20:05:12 -07:00
local file = assert ( FILE_CACHE [ filename ] , " Could not find file: " .. tostring ( filename ) )
2018-04-20 16:23:53 -07:00
return self : run_lua ( Lua ( Source ( filename ) , file ) )
2018-01-10 16:22:45 -08:00
end
if filename : match ( " .*%.nom " ) then
if not self.skip_precompiled then
2018-04-11 20:05:12 -07:00
local lua_filename = filename : gsub ( " %.nom$ " , " .lua " )
local file = FILE_CACHE [ lua_filename ]
2018-01-10 16:22:45 -08:00
if file then
2018-04-20 16:23:53 -07:00
return self : run_lua ( Lua ( Source ( filename ) , file ) )
2018-01-10 16:22:45 -08:00
end
end
2018-04-11 20:05:12 -07:00
local file = file or FILE_CACHE [ filename ]
2018-01-10 16:22:45 -08:00
if not file then
2018-01-25 17:34:49 -08:00
error ( " File does not exist: " .. tostring ( filename ) , 0 )
2018-01-10 16:22:45 -08:00
end
2018-04-20 16:23:53 -07:00
return self : run ( Nomsu ( Source ( filename ) , file ) , compile_fn )
2018-01-10 16:22:45 -08:00
else
2018-01-25 17:34:49 -08:00
return error ( " Invalid filetype for " .. tostring ( filename ) , 0 )
2018-01-10 16:22:45 -08:00
end
end ,
2018-02-05 15:34:57 -08:00
use_file = function ( self , filename )
2018-01-12 19:27:59 -08:00
local loaded = self.environment . LOADED
2018-02-02 15:48:28 -08:00
if not loaded [ filename ] then
2018-02-05 15:34:57 -08:00
for i , f in ipairs ( self.use_stack ) do
if f == filename then
local loop
do
local _accum_0 = { }
local _len_0 = 1
for j = i , # self.use_stack do
_accum_0 [ _len_0 ] = self.use_stack [ 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 ( self.use_stack , filename )
2018-02-02 15:48:28 -08:00
loaded [ filename ] = self : run_file ( filename ) or true
2018-01-10 16:22:45 -08:00
end
2018-02-02 15:48:28 -08:00
return loaded [ filename ]
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
if rawget ( FILE_CACHE , lua.source . filename ) == nil then
FILE_CACHE [ lua.source . filename ] = lua_string
2018-04-11 20:05:12 -07:00
end
2018-04-18 15:28:46 -07:00
if rawget ( FILE_CACHE , lua.source ) == nil then
FILE_CACHE [ lua.source ] = lua_string
2018-01-10 20:45:03 -08:00
end
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-04-11 20:05:12 -07:00
tree_to_value = function ( self , tree )
2018-01-25 17:34:49 -08:00
if tree.type == ' Text ' and # tree.value == 1 and type ( tree.value [ 1 ] ) == ' string ' then
return tree.value [ 1 ]
end
2018-04-25 15:37:13 -07:00
local lua = Lua ( tree.source , " return " , tree : as_lua ( self ) , " ; " )
2018-04-18 17:41:40 -07:00
return self : run_lua ( lua )
2017-09-12 21:38:54 -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
2018-01-26 15:18:56 -08:00
_accum_0 [ _len_0 ] = tostring ( self : value_to_nomsu ( k ) ) .. " : " .. tostring ( self : value_to_nomsu ( v ) )
2017-12-04 17:35:47 -08:00
_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
2018-01-25 17:34:49 -08:00
return error ( " Unsupported value_to_nomsu type: " .. tostring ( type ( value ) ) , 0 )
2017-12-04 17:35:47 -08: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 )
2018-02-13 15:17:45 -08:00
if not ( Types.is_node ( tree ) ) then
2017-09-24 20:20:43 -07:00
return
end
local _exp_0 = tree.type
2018-04-17 14:18:23 -07:00
if " List " == _exp_0 or " File " == _exp_0 or " Block " == _exp_0 or " Action " == _exp_0 or " Text " == _exp_0 or " IndexChain " == _exp_0 then
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
2018-04-17 14:18:23 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local e = _list_0 [ _index_0 ]
2018-02-13 15:17:45 -08:00
self : walk_tree ( e.key , depth + 1 )
self : walk_tree ( e.value , depth + 1 )
2018-01-03 00:52:01 -08:00
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 )
2018-01-26 15:03:07 -08:00
io.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
2018-02-13 15:17:45 -08:00
if Types.is_node ( node ) then
2018-01-26 15:03:07 -08:00
print ( tostring ( ( " " ) : rep ( depth ) ) .. tostring ( node.type ) .. " : " )
2018-02-13 15:17:45 -08:00
else
print ( ( " " ) : rep ( depth ) .. repr ( node ) )
2017-09-26 15:27:01 -07:00
end
end
2018-01-26 15:03:07 -08:00
return io.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
2018-02-13 15:17:45 -08:00
if Types.is_node ( node ) then
2017-09-26 15:27:01 -07:00
insert ( bits , ( tostring ( ( " " ) : rep ( depth ) ) .. tostring ( node.type ) .. " : " ) )
2018-02-13 15:17:45 -08:00
else
insert ( bits , ( ( " " ) : rep ( depth ) .. repr ( node ) ) )
2017-09-26 15:27:01 -07:00
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-02-12 14:47:56 -08:00
tree_map = function ( self , tree , fn )
2018-02-13 15:17:45 -08:00
if not ( Types.is_node ( tree ) ) then
2017-09-24 20:20:43 -07:00
return tree
end
2018-02-12 14:47:56 -08:00
local replacement = fn ( tree )
if replacement ~= nil then
return replacement
end
2017-09-24 20:20:43 -07:00
local _exp_0 = tree.type
2018-04-17 14:18:23 -07:00
if " File " == _exp_0 or " Nomsu " == _exp_0 or " Block " == _exp_0 or " List " == _exp_0 or " Action " == _exp_0 or " Text " == _exp_0 or " IndexChain " == _exp_0 then
2018-02-12 14:47:56 -08:00
local new_values , is_changed = { } , false
for i , old_value in ipairs ( tree.value ) do
local new_value = type ( old_value ) ~= " string " and self : tree_map ( old_value , fn ) or nil
if new_value ~= nil and new_value ~= old_value then
is_changed = true
new_values [ i ] = new_value
else
new_values [ i ] = old_value
end
2017-09-24 20:20:43 -07:00
end
2018-02-12 14:47:56 -08:00
if is_changed then
2018-04-11 20:05:12 -07:00
return tree : with_value ( Tuple ( table.unpack ( new_values ) ) )
2017-09-24 20:20:43 -07:00
end
2018-01-03 00:52:01 -08:00
elseif " Dict " == _exp_0 then
2018-02-12 14:47:56 -08:00
local new_values , is_changed = { } , false
2018-01-03 00:52:01 -08:00
for i , e in ipairs ( tree.value ) do
2018-02-13 15:17:45 -08:00
local new_key = self : tree_map ( e.key , fn )
local new_value = self : tree_map ( e.value , fn )
if ( new_key ~= nil and new_key ~= e.key ) or ( new_value ~= nil and new_value ~= e.value ) then
2018-02-12 14:47:56 -08:00
is_changed = true
2018-02-13 15:17:45 -08:00
new_values [ i ] = DictEntry ( new_key , new_value )
2018-02-12 14:47:56 -08:00
else
new_values [ i ] = e
end
2018-01-03 00:52:01 -08:00
end
2018-02-12 14:47:56 -08:00
if is_changed then
2018-04-11 20:05:12 -07:00
return tree : with_value ( Tuple ( table.unpack ( new_values ) ) )
2018-01-03 00:52:01 -08:00
end
2017-09-24 20:20:43 -07:00
elseif nil == _exp_0 then
2018-02-12 14:47:56 -08:00
error ( " Invalid tree: " .. tostring ( repr ( tree ) ) )
2017-09-24 20:20:43 -07:00
end
return tree
end ,
2018-02-12 14:47:56 -08:00
tree_with_replaced_vars = function ( self , tree , replacements )
return self : tree_map ( tree , function ( t )
if t.type == " Var " then
2018-04-24 20:17:19 -07:00
local id = tostring ( t : as_lua ( self ) )
2018-02-12 14:47:56 -08:00
if replacements [ id ] ~= nil then
return replacements [ id ]
end
end
end )
end ,
2018-03-06 15:29:44 -08:00
tree_to_stub = function ( self , tree )
2018-04-17 14:18:23 -07:00
if tree.type ~= " Action " then
2018-02-08 16:22:57 -08:00
error ( " Tried to get stub from non-functioncall tree: " .. tostring ( tree.type ) , 0 )
end
return concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
2018-04-17 14:18:23 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local t = _list_0 [ _index_0 ]
2018-02-08 16:22:57 -08:00
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " " )
2018-03-06 15:29:44 -08:00
end ,
tree_to_named_stub = function ( self , tree )
2018-04-17 14:18:23 -07:00
if tree.type ~= " Action " then
2018-02-08 16:22:57 -08:00
error ( " Tried to get stub from non-functioncall tree: " .. tostring ( tree.type ) , 0 )
end
return concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
2018-04-17 14:18:23 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local t = _list_0 [ _index_0 ]
2018-02-08 16:22:57 -08:00
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " .. tostring ( t.value ) )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " " )
2018-03-06 15:29:44 -08:00
end ,
2018-01-25 17:34:49 -08:00
get_stubs_from_signature = function ( self , signature )
if type ( signature ) ~= ' table ' or signature.type then
error ( " Invalid signature: " .. tostring ( repr ( signature ) ) , 0 )
end
local stubs = { }
for i , alias in ipairs ( signature ) do
if type ( alias ) ~= ' string ' then
error ( " Expected entries in signature to be strings, not " .. tostring ( type ( alias ) ) .. " s like: " .. tostring ( repr ( alias ) ) .. " \n signature: " .. tostring ( repr ( signature ) ) , 0 )
end
stubs [ i ] = stub_pattern : match ( alias )
if not ( stubs [ i ] ) then
error ( " Failed to match stub pattern on alias: " .. tostring ( repr ( alias ) ) )
end
2017-09-12 21:38:54 -07:00
end
2018-01-25 17:34:49 -08:00
return stubs
2017-09-12 21:38:54 -07:00
end ,
2018-01-25 17:34:49 -08:00
get_args_from_signature = function ( self , signature )
if type ( signature ) ~= ' table ' or signature.type then
error ( " Invalid signature: " .. tostring ( repr ( signature ) ) , 0 )
end
local stub_args = { }
for i , alias in ipairs ( signature ) do
if type ( alias ) ~= ' string ' then
error ( " Invalid type for signature: " .. tostring ( type ( alias ) ) .. " for: \n " .. tostring ( repr ( alias ) ) , 0 )
2017-10-02 19:00:58 -07:00
end
2018-01-25 17:34:49 -08:00
local args = var_pattern : match ( alias )
if not ( args ) then
error ( " Failed to match arg pattern on alias: " .. tostring ( repr ( alias ) ) , 0 )
2017-10-02 19:00:58 -07:00
end
2018-04-11 20:05:12 -07:00
for j = 1 , # args do
2018-01-25 17:34:49 -08:00
args [ j ] = self : var_to_lua_identifier ( args [ j ] )
end
stub_args [ i ] = args
2017-10-02 19:00:58 -07:00
end
2018-01-25 17:34:49 -08:00
return stub_args
2017-10-02 19:00:58 -07:00
end ,
2017-09-21 00:10:26 -07:00
var_to_lua_identifier = function ( self , var )
2018-02-13 15:17:45 -08:00
if Types.Var : is_instance ( var ) then
2017-09-24 20:20:43 -07:00
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-09-12 21:38:54 -07:00
initialize_core = function ( self )
2018-01-25 17:34:49 -08:00
local get_line_no
get_line_no = function ( )
2018-04-08 18:11:44 -07:00
return " nomsu.moon: " .. tostring ( debug_getinfo ( 2 ) . currentline )
2018-01-25 17:34:49 -08:00
end
2018-01-12 16:33:11 -08:00
local nomsu = self
2018-04-13 14:54:35 -07:00
self : define_compile_action ( " immediately %block " , get_line_no ( ) , function ( self , _block )
2018-04-25 15:37:13 -07:00
local lua = _block : as_lua ( nomsu )
2018-04-17 14:18:23 -07:00
lua : convert_to_statements ( )
2018-04-11 20:05:12 -07:00
lua : declare_locals ( )
nomsu : run_lua ( lua )
2018-04-24 20:17:19 -07:00
return Lua ( self.source , " if IMMEDIATE then \n " , lua , " \n end " )
2018-04-13 14:54:35 -07:00
end )
2018-04-20 16:23:53 -07:00
local add_lua_bits
add_lua_bits = function ( lua , code )
if code.type ~= " Text " then
lua : append ( " , " , code : as_lua ( nomsu ) )
return
2018-04-13 14:54:35 -07:00
end
2018-04-20 16:23:53 -07:00
local _list_0 = code.value
2018-04-17 14:18:23 -07:00
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-04-25 15:37:13 -07:00
local bit_lua = bit : as_lua ( nomsu )
2018-04-13 14:54:35 -07:00
if not ( bit_lua.is_value ) then
local line , src = bit.source : get_line ( ) , bit.source : get_text ( )
2018-04-20 16:23:53 -07:00
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " 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
self : define_compile_action ( " Lua %code " , get_line_no ( ) , function ( self , _code )
local lua = Lua.Value ( self.source , " Lua( " , tostring ( _code.source ) )
add_lua_bits ( lua , _code )
lua : append ( " ) " )
return lua
end )
self : define_compile_action ( " Lua %source %code " , get_line_no ( ) , function ( self , _source , _code )
local lua = Lua.Value ( self.source , " Lua( " , _source : as_lua ( nomsu ) )
add_lua_bits ( lua , _code )
lua : append ( " ) " )
return lua
end )
self : define_compile_action ( " Lua value %code " , get_line_no ( ) , function ( self , _code )
local lua = Lua.Value ( self.source , " Lua.Value( " , tostring ( _code.source ) )
add_lua_bits ( lua , _code )
2018-04-13 14:54:35 -07:00
lua : append ( " ) " )
return lua
end )
2018-04-17 14:18:23 -07:00
self : define_compile_action ( " Lua value %source %code " , get_line_no ( ) , function ( self , _source , _code )
2018-04-20 16:23:53 -07:00
local lua = Lua.Value ( self.source , " Lua.Value( " , _source : as_lua ( nomsu ) )
add_lua_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-04-13 14:54:35 -07:00
self : define_compile_action ( " lua> %code " , get_line_no ( ) , function ( self , _code )
2018-04-11 20:05:12 -07:00
if _code.type ~= " Text " then
2018-04-25 15:37:13 -07:00
return Lua.Value ( self.source , " nomsu:run_lua(Lua( " , repr ( _code.source ) , " , " , repr ( tostring ( _code : as_lua ( nomsu ) ) ) , " )) " )
2018-04-11 20:05:12 -07:00
end
local lua = Lua ( _code.source )
2018-04-17 14:18:23 -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-04-25 15:37:13 -07:00
local bit_lua = bit : as_lua ( nomsu )
2018-04-11 21:07:13 -07:00
if not ( bit_lua.is_value ) then
2018-04-11 20:05:12 -07:00
local line , src = bit.source : get_line ( ) , bit.source : get_text ( )
2018-02-08 16:22:57 -08:00
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " 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-01-08 18:53:57 -08:00
end )
2018-04-13 14:54:35 -07:00
self : define_compile_action ( " =lua %code " , get_line_no ( ) , function ( self , _code )
2018-04-11 20:05:12 -07:00
if _code.type ~= " Text " then
2018-04-25 15:37:13 -07:00
return Lua.Value ( self.source , " nomsu:run_lua(Lua( " , repr ( _code.source ) , " , " , repr ( tostring ( _code : as_lua ( nomsu ) ) ) , " )) " )
2018-01-24 13:13:03 -08:00
end
2018-04-17 14:18:23 -07:00
local lua = Lua.Value ( self.source )
local _list_0 = _code.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
2018-04-11 20:05:12 -07:00
if type ( bit ) == " string " then
lua : append ( bit )
else
2018-04-25 15:37:13 -07:00
local bit_lua = bit : as_lua ( nomsu )
2018-04-11 20:05:12 -07:00
if not ( lua.is_value ) then
local line , src = bit.source : get_line ( ) , bit.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " as a string interpolation value, since it's not an expression. " , 0 )
end
lua : append ( bit_lua )
end
end
return lua
2018-01-08 18:53:57 -08:00
end )
2018-04-13 14:54:35 -07:00
self : define_compile_action ( " !! code location !! " , get_line_no ( ) , function ( self )
2018-04-17 14:18:23 -07:00
return Lua.Value ( self.source , repr ( tostring ( self.source ) ) )
2017-12-30 14:31:07 -08:00
end )
2018-04-11 20:05:12 -07:00
self : define_action ( " run file %filename " , get_line_no ( ) , function ( _filename )
2018-02-06 22:06:39 -08:00
return nomsu : run_file ( _filename )
2018-01-10 16:22:45 -08:00
end )
2018-04-13 14:54:35 -07:00
return self : define_compile_action ( " use %filename " , get_line_no ( ) , function ( self , _filename )
2018-01-12 16:33:11 -08:00
local filename = nomsu : tree_to_value ( _filename )
2018-02-05 15:34:57 -08:00
nomsu : use_file ( filename )
2018-04-17 14:18:23 -07:00
return Lua.Value ( self.source , " nomsu:use_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 )
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-05 15:34:57 -08:00
self.use_stack = { }
2018-02-08 16:22:57 -08:00
self.file_metadata = setmetatable ( { } , {
__mode = " k "
} )
self.action_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-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 )
return error ( " Attempt to run undefined action: " .. tostring ( key ) , 0 )
end
} )
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-04-18 15:45:58 -07:00
_nomsu_chunk_counter = 0
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-25 17:34:49 -08:00
stub_defs = {
space = ( P ( ' ' ) + P ( ' \n .. ' ) ) ^ 0 ,
word = ( NOMSU_DEFS.ident_char ^ 1 + NOMSU_DEFS.operator ^ 1 ) ,
varname = ( R ( ' az ' , ' AZ ' , ' 09 ' ) + P ( ' _ ' ) + NOMSU_DEFS.utf8_char ) ^ 0
}
stub_pattern = re.compile ( " {~ (%space->'') (('%' (%varname->'')) / %word)? ((%space->' ') (('%' (%varname->'')) / %word))* (%space->'') ~} " , stub_defs )
var_pattern = re.compile ( " {| %space ((('%' {%varname}) / %word) %space)+ |} " , stub_defs )
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 ' )
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 ( )
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
do
local metadata = nomsu.action_metadata [ info.func ]
if metadata then
info.name = metadata.aliases [ 1 ]
2018-04-18 17:41:40 -07:00
local _ = [ = [ filename = if type ( metadata.source ) == ' string '
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-12 20:39:17 -07:00
end
2018-04-11 20:05:12 -07:00
end
2018-04-08 18:11:44 -07:00
end
return info
end
2018-01-11 01:03:52 -08:00
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
2018-04-19 19:43:23 -07:00
local compile_fn = nil
2018-01-11 01:03:52 -08:00
if args.flags [ " -p " ] then
2018-01-26 15:02:09 -08:00
nomsu.environment . print = function ( ) end
2018-04-19 19:43:23 -07:00
compile_fn = function ( code )
2018-04-24 20:17:19 -07:00
return io.output ( ) : write ( " local IMMEDIATE = true; \n " .. tostring ( code ) )
2018-04-19 19:43:23 -07:00
end
2018-01-11 01:03:52 -08:00
elseif args.output then
2018-04-19 19:43:23 -07:00
compile_fn = function ( code )
2018-04-24 20:17:19 -07:00
return io.open ( args.output , ' w ' ) : write ( " local IMMEDIATE = true; \n " .. tostring ( code ) )
2018-04-19 19:43:23 -07:00
end
2018-01-11 01:03:52 -08:00
end
if args.input : match ( " .*%.lua " ) then
2018-04-19 19:43:23 -07:00
dofile ( args.input ) ( nomsu , { } )
2017-10-08 18:23:48 -07:00
else
2018-04-19 19:43:23 -07:00
if args.input == " - " then
nomsu : run ( io.read ( ' a ' ) , compile_fn )
2018-01-11 01:03:52 -08:00
else
2018-04-19 19:43:23 -07:00
nomsu : run_file ( args.input , compile_fn )
2018-01-11 01:03:52 -08:00
end
2017-10-08 18:23:48 -07:00
end
2018-01-11 01:03:52 -08:00
if args.flags [ " -p " ] then
2018-01-26 15:02:09 -08:00
nomsu.environment . print = print
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-04-18 15:45:58 -07:00
nomsu : run ( ' use "core" ' )
2018-01-11 01:03:52 -08:00
while true do
2018-01-30 16:40:05 -08:00
io.write ( colored.bright ( colored.yellow ( " >> " ) ) )
2018-01-11 01:03:52 -08:00
local buff = " "
while true do
local line = io.read ( " *L " )
if line == " \n " or not line then
break
end
2018-01-30 16:40:05 -08:00
line = line : gsub ( " \t " , " " )
2018-01-11 01:03:52 -08:00
buff = buff .. line
2018-01-30 16:40:05 -08:00
io.write ( colored.dim ( colored.yellow ( " .. " ) ) )
2018-01-11 01:03:52 -08:00
end
if # buff == 0 then
break
end
2018-04-08 18:11:44 -07:00
local ret
2018-04-18 15:45:58 -07:00
ok , ret = pcall ( nomsu.run , nomsu , buff )
2018-01-11 01:03:52 -08:00
if ok and ret ~= nil then
print ( " = " .. repr ( ret ) )
2018-02-12 14:47:56 -08:00
elseif not ok then
print ( colored.bright ( colored.red ( ret ) ) )
2018-01-11 01:03:52 -08:00
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: " )
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 )
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
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
do
2018-02-08 16:22:57 -08:00
local metadata = nomsu.action_metadata [ calling_fn.func ]
2018-01-12 19:27:59 -08:00
if metadata then
2018-01-25 17:34:49 -08:00
local filename , start , stop = metadata.source : match ( " ([^:]*):([0-9]*),([0-9]*) " )
if filename then
2018-04-11 20:05:12 -07:00
local file = FILE_CACHE [ filename ]
2018-01-25 17:34:49 -08:00
local line_no = 1
for _ in file : sub ( 1 , tonumber ( start ) ) : gmatch ( " \n " ) do
line_no = line_no + 1
end
local offending_statement = file : sub ( tonumber ( start ) , tonumber ( stop ) )
if # offending_statement > 50 then
offending_statement = offending_statement : sub ( 1 , 50 ) .. " ... "
end
offending_statement = colored.red ( offending_statement )
line = colored.yellow ( filename .. " : " .. tostring ( line_no ) .. " \n " .. offending_statement )
else
line = colored.yellow ( metadata.source )
end
2018-01-12 19:27:59 -08:00
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
2018-01-27 16:39:56 -08:00
if calling_fn.short_src == " ./nomsu.moon " and line_table 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-04-25 16:04:46 -07:00
require ( ' ldt ' ) . guard ( run )
2017-09-12 21:38:54 -07:00
end
2017-09-13 16:22:04 -07:00
return NomsuCompiler