2017-09-12 21:38:54 -07:00
local re = require ( ' re ' )
local lpeg = require ( ' lpeg ' )
local utils = require ( ' utils ' )
2017-09-21 21:11:13 -07:00
local repr = utils.repr
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 )
return colors [ color ] .. msg .. colors.reset
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
2017-12-04 17:35:47 -08:00
if _VERSION == " Lua 5.1 " then
local xp = xpcall
local xpcall
xpcall = function ( f , errhandler , ... )
local args = {
n = select ( " # " , ... ) ,
...
}
return xp ( function ( ... )
return f ( unpack ( args , 1 , args.n ) )
end ) , errhandler
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
2017-09-21 04:51:02 -07:00
local STRING_ESCAPES = {
n = " \n " ,
t = " \t " ,
b = " \b " ,
a = " \a " ,
v = " \v " ,
f = " \f " ,
r = " \r "
}
2017-09-22 00:03:32 -07:00
local indent_stack = {
0
}
2017-12-13 16:29:15 -08:00
local indent_patt = P ( function ( self , start )
local spaces = self : match ( " [ \t ]* " , start )
2017-09-22 00:03:32 -07:00
if # spaces > indent_stack [ # indent_stack ] then
insert ( indent_stack , # spaces )
2017-12-13 16:29:15 -08:00
return start + # spaces
2017-09-22 00:03:32 -07:00
end
2017-12-13 16:29:15 -08:00
end )
local dedent_patt = P ( function ( self , start )
local spaces = self : match ( " [ \t ]* " , start )
2017-09-22 00:03:32 -07:00
if # spaces < indent_stack [ # indent_stack ] then
remove ( indent_stack )
2017-12-13 16:29:15 -08:00
return start
2017-09-22 00:03:32 -07:00
end
2017-12-13 16:29:15 -08:00
end )
local nodent_patt = P ( function ( self , start )
local spaces = self : match ( " [ \t ]* " , start )
2017-09-22 00:03:32 -07:00
if # spaces == indent_stack [ # indent_stack ] then
2017-12-13 16:29:15 -08:00
return start + # spaces
2017-09-22 00:03:32 -07:00
end
2017-12-13 16:29:15 -08:00
end )
local gt_nodent_patt = P ( function ( self , start )
local spaces = self : match ( " [ \t ]* " , start )
if # spaces >= indent_stack [ # indent_stack ] + 4 then
return start + indent_stack [ # indent_stack ] + 4
end
end )
2017-10-13 16:10:47 -07:00
local nomsu = [ = [ file <- ( { { | shebang ?
2017-09-24 20:20:43 -07:00
( ignored_line % nl ) *
statements ( nodent statements ) *
( % nl ignored_line ) * % nl ?
2017-12-08 15:37:36 -08:00
( ( { . + } ( " " -> " Parse error " ) ) => error ) ? | } } ) -> File
2017-09-22 00:03:32 -07:00
shebang <- " #! " [ ^% nl ] * % nl
2017-09-24 20:20:43 -07:00
inline_statements <- inline_statement ( semicolon inline_statement ) *
noeol_statements <- ( inline_statement semicolon ) * noeol_statement
statements <- ( inline_statement semicolon ) * statement
statement <- functioncall / expression
noeol_statement <- noeol_functioncall / noeol_expression
inline_statement <- inline_functioncall / inline_expression
2017-09-22 00:03:32 -07:00
2017-10-13 19:41:58 -07:00
inline_thunk <- ( { { | " { " % ws ? inline_statements % ws ? " } " | } } ) -> Thunk
2017-10-02 19:00:58 -07:00
eol_thunk <- ( { { | " : " % ws ? noeol_statements eol | } } ) -> Thunk
indented_thunk <- ( { { | ( " : " / " {..} " ) indent
2017-09-26 15:27:01 -07:00
statements ( nodent statements ) *
2017-10-02 19:00:58 -07:00
( dedent / ( ( { . + } ( " " -> " Error while parsing thunk " ) ) => error ) )
| } } ) -> Thunk
2017-09-22 00:03:32 -07:00
2017-10-13 16:10:47 -07:00
inline_nomsu <- ( { ( " \" inline_expression) }) -> Nomsu
eol_nomsu <- ( { ( " \" noeol_expression) }) -> Nomsu
indented_nomsu <- ( { ( " \" expression) }) -> Nomsu
2017-09-22 00:03:32 -07:00
2017-10-02 19:00:58 -07:00
inline_expression <- number / variable / inline_string / inline_list / inline_nomsu
2017-10-13 19:41:58 -07:00
/ inline_thunk / ( " ( " % ws ? inline_statement % ws ? " ) " )
2017-10-02 19:00:58 -07:00
noeol_expression <- indented_string / indented_nomsu / indented_list / indented_thunk
/ ( " (..) " indent
statement
( dedent / ( ( { . + } ( " " -> " Error while parsing indented expression " ) ) ) )
) / inline_expression
expression <- eol_thunk / eol_nomsu / noeol_expression
2017-09-22 00:03:32 -07:00
-- Function calls need at least one word in them
2017-10-13 16:10:47 -07:00
inline_functioncall <- ( { ( ' ' => line_no ) { |
2017-12-08 15:39:05 -08:00
( inline_expression % ws ? ) * word ( % ws ? ( inline_expression / word ) ) *
2017-09-24 20:20:43 -07:00
| } } ) -> FunctionCall
2017-10-13 16:10:47 -07:00
noeol_functioncall <- ( { ( ' ' => line_no ) { |
2017-12-08 15:39:05 -08:00
( noeol_expression % ws ? ) * word ( % ws ? ( noeol_expression / word ) ) *
2017-09-24 20:20:43 -07:00
| } } ) -> FunctionCall
2017-10-13 16:10:47 -07:00
functioncall <- ( { ( ' ' => line_no ) { |
2017-12-08 15:39:05 -08:00
( expression ( dotdot / % ws ? ) ) * word ( ( dotdot / % ws ? ) ( expression / word ) ) *
2017-09-22 00:03:32 -07:00
| } } ) -> FunctionCall
2017-12-08 15:37:36 -08:00
word <- ( { { % operator / ( ! number % plain_word ) } } ) -> Word
2017-09-22 00:03:32 -07:00
2017-09-24 20:20:43 -07:00
inline_string <- ( { ' " ' { |
2017-09-29 22:04:03 -07:00
( { ~ ( ( " \\ " -> " \" ) / (' \" ' -> ' " ' ) / (" \n " -> "
" ) / (!string_interpolation [^%nl " ] ) ) + ~ }
2017-09-24 20:20:43 -07:00
/ string_interpolation ) * | } ' " ' } ) -> String
2017-12-13 16:29:15 -08:00
indented_string <- ( { ' ".." ' % ws ? line_comment ? % nl % gt_nodented ? { |
( { ~ ( ( " \\ " -> " \" ) / (%nl+ {~ %gt_nodented -> " " ~}) / [^%nl \ ]) ~} / string_interpolation)*
| } ( ( ! . ) / ( & ( % nl + ! % gt_nodented ) ) / ( ( { . + } ( " " -> " Error while parsing String " ) ) => error ) )
2017-09-24 20:20:43 -07:00
} ) -> String
2017-12-13 16:29:15 -08:00
2017-10-02 19:00:58 -07:00
string_interpolation <- " \" ((noeol_expression dotdot?) / dotdot)
2017-09-22 00:03:32 -07:00
2017-09-24 20:20:43 -07:00
number <- ( { ( ( " - " ? ( ( [ 0 - 9 ] + " . " [ 0 - 9 ] + ) / ( " . " [ 0 - 9 ] + ) / ( [ 0 - 9 ] + ) ) ) -> tonumber ) } ) -> Number
2017-09-22 00:03:32 -07:00
2017-12-08 15:37:36 -08:00
-- Variables can be nameless (i.e. just %) and can't contain operators like apostrophe
2017-10-30 14:08:23 -07:00
-- which is a hack to allow %'s to parse as "%" and "' s" separately
2017-12-08 15:37:36 -08:00
variable <- ( { ( " % " { % plain_word ? } ) } ) -> Var
2017-09-22 00:03:32 -07:00
2017-09-24 20:20:43 -07:00
inline_list <- ( { { |
( " [ " % ws ? ( ( inline_list_item comma ) * inline_list_item comma ? ) ? % ws ? " ] " )
| } } ) -> List
indented_list <- ( { { |
2017-09-22 00:03:32 -07:00
( " [..] " indent
list_line ( nodent list_line ) *
( dedent / ( ( { . + } ( " " -> " Error while parsing list " ) ) => error ) ) )
| } } ) -> List
2017-09-24 20:20:43 -07:00
list_line <- ( inline_list_item comma ) * ( ( inline_list_item % ws ? " , " ) / ( functioncall / expression ) )
inline_list_item <- inline_functioncall / inline_expression
2017-09-22 00:03:32 -07:00
2017-10-02 20:17:52 -07:00
block_comment <- " #.. " [ ^% nl ] * ( % nl ( % ws ? &% nl ) ) * % nl % indented [ ^% nl ] + ( % nl ( ( % ws ? ( ! . / &% nl ) ) / ( ! % dedented [ ^% nl ] + ) ) ) *
2017-09-22 00:03:32 -07:00
line_comment <- " # " [ ^% nl ] *
eol <- % ws ? line_comment ? ( ! . / &% nl )
ignored_line <- ( % nodented ( block_comment / line_comment ) ) / ( % ws ? ( ! . / &% nl ) )
2017-10-02 20:17:52 -07:00
indent <- eol ( % nl ignored_line ) * % nl % indented ( ( block_comment / line_comment ) ( % nl ignored_line ) * nodent ) ?
2017-09-22 00:03:32 -07:00
nodent <- eol ( % nl ignored_line ) * % nl % nodented
dedent <- eol ( % nl ignored_line ) * ( ( ( ! . ) &% dedented ) / ( & ( % nl % dedented ) ) )
2017-09-24 20:20:43 -07:00
comma <- % ws ? " , " % ws ?
semicolon <- % ws ? " ; " % ws ?
2017-09-22 00:03:32 -07:00
dotdot <- nodent " .. " % ws ?
] = ]
2017-10-13 16:10:47 -07:00
local CURRENT_FILE = nil
2017-09-22 00:03:32 -07:00
local whitespace = S ( " \t " ) ^ 1
2017-12-08 15:37:36 -08:00
local operator = S ( " '~`!@$^&*-+=|<>?/ " ) ^ 1
local utf8_continuation = R ( " \128 \191 " )
local utf8_char = ( R ( " \194 \223 " ) * utf8_continuation + R ( " \224 \239 " ) * utf8_continuation * utf8_continuation + R ( " \240 \244 " ) * utf8_continuation * utf8_continuation * utf8_continuation )
local plain_word = ( R ( ' az ' , ' AZ ' , ' 09 ' ) + S ( " _ " ) + utf8_char ) ^ 1
2017-09-22 00:03:32 -07:00
local defs = {
ws = whitespace ,
nl = P ( " \n " ) ,
2017-09-24 20:20:43 -07:00
tonumber = tonumber ,
2017-12-08 15:37:36 -08:00
operator = operator ,
plain_word = plain_word ,
2017-12-13 16:29:15 -08:00
indented = indent_patt ,
nodented = nodent_patt ,
dedented = dedent_patt ,
gt_nodented = gt_nodent_patt ,
2017-10-13 16:10:47 -07:00
line_no = function ( src , pos )
local line_no = 1
for _ in src : sub ( 1 , pos ) : gmatch ( " \n " ) do
line_no = line_no + 1
end
return pos , tostring ( CURRENT_FILE ) .. " : " .. tostring ( line_no )
end ,
FunctionCall = function ( src , line_no , value , errors )
2017-12-04 17:35:47 -08:00
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 ) ( ) , " " )
2017-10-13 16:10:47 -07:00
return {
type = " FunctionCall " ,
src = src ,
line_no = line_no ,
value = value ,
2017-12-04 17:35:47 -08:00
errors = errors ,
stub = stub
2017-10-13 16:10:47 -07:00
}
end ,
2017-09-22 00:03:32 -07:00
error = function ( src , pos , errors , err_msg )
local line_no = 1
for _ in src : sub ( 1 , -# errors ) : gmatch ( " \n " ) do
line_no = line_no + 1
end
local err_pos = # src - # errors + 1
if errors : sub ( 1 , 1 ) == " \n " then
err_pos = err_pos + # errors : match ( " [ \t ]* " , 2 )
end
local start_of_err_line = err_pos
while src : sub ( start_of_err_line , start_of_err_line ) ~= " \n " and start_of_err_line > 1 do
start_of_err_line = start_of_err_line - 1
end
local start_of_prev_line = start_of_err_line - 1
while src : sub ( start_of_prev_line , start_of_prev_line ) ~= " \n " and start_of_prev_line > 1 do
start_of_prev_line = start_of_prev_line - 1
end
local prev_line , err_line , next_line
prev_line , err_line , next_line = src : match ( " ([^ \n ]*) \n ([^ \n ]*) \n ([^ \n ]*) " , start_of_prev_line + 1 )
local pointer = ( " - " ) : rep ( err_pos - start_of_err_line + 0 ) .. " ^ "
2017-10-13 16:10:47 -07:00
return error ( " \n " .. tostring ( err_msg or " Parse error " ) .. " in " .. tostring ( CURRENT_FILE ) .. " on line " .. tostring ( line_no ) .. " : \n \n " .. tostring ( prev_line ) .. " \n " .. tostring ( err_line ) .. " \n " .. tostring ( pointer ) .. " \n " .. tostring ( next_line ) .. " \n " )
2017-09-22 00:03:32 -07:00
end
}
setmetatable ( defs , {
__index = function ( t , key )
2017-09-22 00:51:53 -07:00
do
local _with_0
_with_0 = function ( src , value , errors )
return {
type = key ,
src = src ,
value = value ,
errors = errors
}
end
t [ key ] = _with_0
local _ = nil
return _with_0
2017-09-22 00:03:32 -07:00
end
end
} )
nomsu = re.compile ( nomsu , defs )
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 ,
2017-10-02 19:00:58 -07:00
def = function ( self , signature , thunk , src , is_macro )
if is_macro == nil then
is_macro = 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
2017-12-08 15:37:36 -08:00
if self.debug then
self : write ( colored.magenta ( " Defined rule " .. tostring ( repr ( signature ) ) ) )
end
2017-10-02 19:00:58 -07:00
assert ( type ( thunk ) == ' function ' , " Bad thunk: " .. tostring ( repr ( thunk ) ) )
local canonical_args = nil
2017-12-04 17:35:47 -08:00
local canonical_escaped_args = nil
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
local def = {
thunk = thunk ,
src = src ,
is_macro = is_macro ,
aliases = { } ,
def_number = self.__class . def_number ,
defs = self.defs
}
2017-12-09 15:34:52 -08:00
local where_defs_go = ( getmetatable ( self.defs ) or { } ) . __newindex or self.defs
2017-10-13 16:10:47 -07:00
for _index_0 = 1 , # signature do
local _des_0 = signature [ _index_0 ]
2017-12-04 17:35:47 -08:00
local stub , arg_names , escaped_args
stub , arg_names , escaped_args = _des_0 [ 1 ] , _des_0 [ 2 ] , _des_0 [ 3 ]
2017-10-02 19:00:58 -07:00
assert ( stub , " NO STUB FOUND: " .. tostring ( repr ( signature ) ) )
if self.debug then
self : writeln ( tostring ( colored.bright ( " DEFINING RULE: " ) ) .. " " .. tostring ( colored.underscore ( colored.magenta ( repr ( stub ) ) ) ) .. " " .. tostring ( colored.bright ( " WITH ARGS " ) ) .. " " .. tostring ( colored.dim ( repr ( arg_names ) ) ) )
end
for i = 1 , # arg_names - 1 do
for j = i + 1 , # arg_names do
if arg_names [ i ] == arg_names [ j ] then
self : error ( " Duplicate argument in function " .. tostring ( stub ) .. " : ' " .. tostring ( arg_names [ i ] ) .. " ' " )
end
2017-09-24 20:20:43 -07:00
end
end
2017-10-02 19:00:58 -07:00
if canonical_args then
assert ( utils.equivalent ( utils.set ( arg_names ) , canonical_args ) , " Mismatched args " )
else
canonical_args = utils.set ( arg_names )
end
2017-12-04 17:35:47 -08:00
if canonical_escaped_args then
assert ( utils.equivalent ( escaped_args , canonical_escaped_args ) , " Mismatched escaped args " )
else
canonical_escaped_args = escaped_args
def.escaped_args = escaped_args
end
insert ( def.aliases , stub )
local stub_def = setmetatable ( {
2017-10-02 19:00:58 -07:00
stub = stub ,
2017-09-26 15:27:01 -07:00
arg_names = arg_names ,
2017-12-04 17:35:47 -08:00
escaped_args = escaped_args
} , {
__index = def
} )
rawset ( where_defs_go , stub , stub_def )
2017-09-18 22:41:50 -07:00
end
2017-09-21 21:11:13 -07:00
end ,
2017-10-02 19:00:58 -07:00
defmacro = function ( self , signature , thunk , src )
return self : def ( signature , thunk , src , true )
2017-09-21 21:11:13 -07:00
end ,
2017-12-04 17:35:47 -08:00
scoped = function ( self , thunk )
local old_defs = self.defs
2017-12-09 15:34:52 -08:00
local new_defs = {
[ " #vars " ] = setmetatable ( { } , {
__index = self.defs [ " #vars " ]
} ) ,
[ " #loaded_files " ] = setmetatable ( { } , {
__index = self.defs [ " #loaded_files " ]
} )
}
self.defs = setmetatable ( new_defs , {
2017-12-04 17:35:47 -08:00
__index = old_defs
} )
local ok , ret1 , ret2 = pcall ( thunk , self )
self.defs = old_defs
if not ok then
self : error ( ret1 )
end
return ret1 , ret2
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
after = 0
end
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-10-13 16:10:47 -07:00
call = function ( self , stub , line_no , ... )
2017-09-26 15:27:01 -07:00
local def = self.defs [ stub ]
2017-10-13 16:10:47 -07:00
if def and def.is_macro and self.callstack [ # self.callstack ] ~= " #macro " then
self : error ( " Attempt to call macro at runtime: " .. tostring ( stub ) .. " \n This can be caused by using a macro in a function that is defined before the macro. " )
end
insert ( self.callstack , {
stub ,
line_no
} )
2017-12-04 17:35:47 -08:00
if not ( def ) then
2017-09-26 15:27:01 -07:00
self : error ( " Attempt to call undefined function: " .. tostring ( stub ) )
2017-09-21 21:11:13 -07:00
end
2017-10-13 14:15:02 -07:00
if not ( def.is_macro ) then
self : assert_permission ( stub )
2017-09-21 21:11:13 -07:00
end
2017-09-26 15:27:01 -07:00
local thunk , arg_names
thunk , arg_names = def.thunk , def.arg_names
local args
2017-09-21 21:11:13 -07:00
do
local _tbl_0 = { }
2017-09-26 15:27:01 -07:00
for i , name in ipairs ( arg_names ) do
2017-09-21 21:11:13 -07:00
_tbl_0 [ name ] = select ( i , ... )
end
args = _tbl_0
end
if self.debug then
2017-09-28 17:49:15 -07:00
self : write ( tostring ( colored.bright ( " CALLING " ) ) .. " " .. tostring ( colored.magenta ( colored.underscore ( stub ) ) ) .. " " )
self : writeln ( tostring ( colored.bright ( " WITH ARGS: " ) ) .. " " .. tostring ( colored.dim ( repr ( args ) ) ) )
2017-09-21 21:11:13 -07:00
end
2017-12-04 17:35:47 -08:00
local old_defs
old_defs , self.defs = self.defs , def.defs
2017-09-21 21:11:13 -07:00
local rets = {
2017-09-24 20:20:43 -07:00
thunk ( self , args )
2017-09-21 21:11:13 -07:00
}
2017-12-04 17:35:47 -08:00
self.defs = old_defs
2017-09-21 21:11:13 -07:00
remove ( self.callstack )
return unpack ( rets )
end ,
2017-10-13 14:15:02 -07:00
run_macro = function ( self , tree )
2017-10-13 16:10:47 -07: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
2017-09-28 17:49:15 -07:00
if self.debug then
2017-12-04 17:35:47 -08:00
self : write ( tostring ( colored.bright ( " RUNNING MACRO " ) ) .. " " .. tostring ( colored.underscore ( colored.magenta ( tree.stub ) ) ) .. " " )
2017-09-28 17:49:15 -07:00
self : writeln ( tostring ( colored.bright ( " WITH ARGS: " ) ) .. " " .. tostring ( colored.dim ( repr ( args ) ) ) )
end
2017-09-24 20:20:43 -07:00
insert ( self.callstack , " #macro " )
2017-12-04 17:35:47 -08:00
local expr , statement = self : call ( tree.stub , tree.line_no , unpack ( args ) )
2017-09-21 21:11:13 -07:00
remove ( self.callstack )
2017-09-24 20:20:43 -07:00
return expr , statement
2017-09-21 21:11:13 -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
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 ,
indent = function ( self , code )
return ( code : gsub ( " \n " , " \n " ) )
end ,
2017-10-13 14:15:02 -07:00
assert_permission = function ( self , stub )
local fn_def = self.defs [ stub ]
if not ( fn_def ) then
self : error ( " Undefined function: " .. tostring ( fn_name ) )
end
local whiteset = fn_def.whiteset
if whiteset == nil then
return true
end
local _list_0 = self.callstack
for _index_0 = 1 , # _list_0 do
local caller = _list_0 [ _index_0 ]
2017-10-20 15:07:57 -07:00
if caller ~= " #macro " and whiteset [ caller [ 1 ] ] then
2017-10-13 14:15:02 -07:00
return true
end
end
return self : error ( " You do not have the authority to call: " .. tostring ( stub ) )
end ,
2017-09-22 00:27:27 -07:00
check_permission = function ( self , fn_def )
if getmetatable ( fn_def ) ~= functiondef_mt then
local fn_name = fn_def
fn_def = self.defs [ fn_name ]
if fn_def == nil then
self : error ( " Undefined function: " .. tostring ( fn_name ) )
end
2017-09-12 21:38:54 -07:00
end
2017-09-22 00:27:27 -07:00
local whiteset = fn_def.whiteset
if whiteset == nil then
2017-09-21 21:11:13 -07:00
return true
end
local _list_0 = self.callstack
for _index_0 = 1 , # _list_0 do
local caller = _list_0 [ _index_0 ]
2017-10-20 15:07:57 -07:00
if caller ~= " #macro " and whiteset [ caller [ 1 ] ] then
2017-09-21 21:11:13 -07:00
return true
end
end
return false
2017-09-12 21:38:54 -07:00
end ,
2017-09-20 04:21:46 -07:00
parse = function ( self , str , filename )
2017-09-12 21:38:54 -07:00
if self.debug then
2017-09-28 17:49:15 -07:00
self : writeln ( tostring ( colored.bright ( " PARSING: " ) ) .. " \n " .. tostring ( str ) )
2017-09-12 21:38:54 -07:00
end
2017-09-24 20:20:43 -07:00
str = str : gsub ( " \r " , " " )
2017-10-13 16:10:47 -07:00
local old_file = CURRENT_FILE
2017-09-22 00:03:32 -07:00
local old_indent_stack
old_indent_stack , indent_stack = indent_stack , {
2017-09-20 04:21:46 -07:00
0
}
2017-10-13 16:10:47 -07:00
CURRENT_FILE = filename
2017-09-22 00:03:32 -07:00
local tree = nomsu : match ( str )
indent_stack = old_indent_stack
2017-10-13 16:10:47 -07:00
CURRENT_FILE = old_file
2017-09-24 20:20:43 -07:00
assert ( tree , " Failed to parse: " .. tostring ( str ) )
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 ,
2017-12-14 16:41:22 -08:00
run = function ( self , src , filename , vars , max_operations , output_file )
2017-10-10 00:52:07 -07:00
if vars == nil then
vars = { }
end
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
return nil , " " , vars
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 ( )
2017-10-09 04:31:41 -07:00
return self : error ( " Execution quota exceeded. Your code took too long. " )
end
debug.sethook ( timeout , " " , max_operations )
end
2017-09-24 20:20:43 -07:00
local tree = self : parse ( src , filename )
assert ( tree , " Tree failed to compile: " .. tostring ( src ) )
2017-09-26 15:27:01 -07:00
assert ( tree.type == " File " , " Attempt to run non-file: " .. tostring ( tree.type ) )
2017-09-24 20:20:43 -07:00
local buffer = { }
local return_value = nil
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local statement = _list_0 [ _index_0 ]
2017-09-26 15:27:01 -07:00
if self.debug then
2017-09-28 17:49:15 -07:00
self : writeln ( tostring ( colored.bright ( " RUNNING NOMSU: " ) ) .. " \n " .. tostring ( colored.bright ( colored.yellow ( statement.src ) ) ) )
self : writeln ( colored.bright ( " PARSED TO TREE: " ) )
2017-09-26 15:27:01 -07:00
self : print_tree ( statement )
end
2017-12-08 15:37:36 -08:00
local ok , expr , statements = pcall ( self.tree_to_lua , self , statement , filename )
2017-09-24 20:20:43 -07:00
if not ok then
2017-10-12 14:39:49 -07:00
self : errorln ( tostring ( colored.red ( " Error occurred in statement: " ) ) .. " \n " .. tostring ( colored.bright ( colored.yellow ( statement.src ) ) ) )
2017-12-04 17:35:47 -08:00
error ( expr )
2017-09-24 20:20:43 -07:00
end
2017-10-13 18:09:46 -07:00
local code_for_statement = ( [ [ return ( function ( nomsu , vars )
% s
return % s ;
end ) ; ] ] ) : format ( statements or " " , expr or " ret " )
2017-12-14 16:41:22 -08:00
if output_file then
if statements and # statements > 0 then
output_file : write ( " lua> \" .. \" \n " .. tostring ( self : indent ( statements : gsub ( " \\ " , " \\ \\ " ) ) ) .. " \n " )
end
if expr and # expr > 0 then
output_file : write ( " =lua \" .. \" \n " .. tostring ( self : indent ( expr : gsub ( " \\ " , " \\ \\ " ) ) ) .. " \n " )
end
end
2017-09-24 20:20:43 -07:00
if self.debug then
2017-09-28 17:49:15 -07:00
self : writeln ( tostring ( colored.bright ( " RUNNING LUA: " ) ) .. " \n " .. tostring ( colored.blue ( colored.bright ( code_for_statement ) ) ) )
2017-09-24 20:20:43 -07:00
end
local lua_thunk , err = load ( code_for_statement )
if not lua_thunk then
2017-10-02 19:00:58 -07:00
local n = 1
local fn
fn = function ( )
n = n + 1
return ( " \n %-3d| " ) : format ( n )
end
local code = " 1 | " .. code_for_statement : gsub ( " \n " , fn )
2017-10-13 18:09:46 -07:00
error ( " Failed to compile generated code: \n " .. tostring ( colored.bright ( colored.blue ( colored.onblack ( code ) ) ) ) .. " \n \n " .. tostring ( err ) .. " \n \n Produced by statement: \n " .. tostring ( colored.bright ( colored.yellow ( statement.src ) ) ) )
2017-09-24 20:20:43 -07:00
end
local run_statement = lua_thunk ( )
local ret
ok , ret = pcall ( run_statement , self , vars )
if expr then
return_value = ret
end
if not ok then
2017-10-12 14:39:49 -07:00
self : errorln ( tostring ( colored.red ( " Error occurred in statement: " ) ) .. " \n " .. tostring ( colored.yellow ( statement.src ) ) )
self : errorln ( debug.traceback ( ) )
2017-12-04 17:35:47 -08:00
error ( ret )
2017-09-24 20:20:43 -07:00
end
2017-11-01 16:49:11 -07:00
if statements then
insert ( buffer , statements )
end
if expr then
insert ( buffer , " ret = " .. tostring ( expr ) .. " ; " )
end
2017-09-24 20:20:43 -07:00
end
2017-10-10 00:52:07 -07:00
if max_operations then
debug.sethook ( )
end
2017-10-13 18:09:46 -07:00
local lua_code = ( [ [ return ( function ( nomsu , vars )
local ret ;
% s
return ret ;
2017-11-01 16:49:11 -07:00
end ) ; ] ] ) : format ( concat ( buffer , " \n " ) )
2017-10-10 00:52:07 -07:00
return return_value , lua_code , vars
2017-09-24 20:20:43 -07:00
end ,
2017-12-08 15:37:36 -08:00
tree_to_value = function ( self , tree , vars , filename )
local code = " return (function(nomsu, vars) \n return " .. tostring ( self : tree_to_lua ( tree , filename ) ) .. " ; \n end); "
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
2017-09-12 21:38:54 -07:00
local lua_thunk , err = load ( code )
if not lua_thunk then
2017-10-13 18:09:46 -07:00
self : 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
2017-09-14 02:41:10 -07:00
return ( lua_thunk ( ) ) ( self , vars or { } )
2017-09-12 21:38:54 -07:00
end ,
2017-10-22 18:40:49 -07:00
tree_to_nomsu = function ( self , tree , force_inline )
if force_inline == nil then
force_inline = false
end
assert ( tree , " No tree provided. " )
if not tree.type then
self : errorln ( debug.traceback ( ) )
self : error ( " Invalid tree: " .. tostring ( repr ( tree ) ) )
end
local _exp_0 = tree.type
if " File " == _exp_0 then
return concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = self : tree_to_nomsu ( v , force_inline )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " \n " ) , false
elseif " Nomsu " == _exp_0 then
local inside , inline = self : tree_to_nomsu ( tree.value , force_inline )
return " \\ " .. tostring ( inside ) , inline
elseif " Thunk " == _exp_0 then
if force_inline then
return " { " .. tostring ( concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = self : tree_to_nomsu ( v , true )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " ; " ) ) , true
else
2017-12-04 17:35:47 -08:00
return " : " .. self : indent ( " \n " .. concat ( ( function ( )
2017-10-22 18:40:49 -07:00
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = self : tree_to_nomsu ( v )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " \n " ) ) , false
end
elseif " FunctionCall " == _exp_0 then
local buff = " "
local sep = " "
local inline = true
2017-12-04 17:35:47 -08:00
local line_len = 0
2017-10-22 18:40:49 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local arg = _list_0 [ _index_0 ]
local arg_inline
nomsu , arg_inline = self : tree_to_nomsu ( arg , force_inline )
2017-12-04 17:35:47 -08:00
if sep == " " and line_len + # nomsu > 80 then
sep = " \n .. "
end
if not ( sep == " " and not arg_inline and nomsu : sub ( 1 , 1 ) == " : " ) then
buff = buff .. sep
end
2017-10-22 18:40:49 -07:00
if arg_inline then
sep = " "
2017-12-04 17:35:47 -08:00
line_len = line_len + ( 1 + # nomsu )
2017-10-22 18:40:49 -07:00
else
2017-12-04 17:35:47 -08:00
line_len = 0
2017-10-22 18:40:49 -07:00
inline = false
sep = " \n .. "
end
if arg.type == ' FunctionCall ' then
if arg_inline then
buff = buff .. " ( " .. tostring ( nomsu ) .. " ) "
else
2017-12-04 17:35:47 -08:00
buff = buff .. " (..) \n " .. tostring ( self : indent ( nomsu ) )
2017-10-22 18:40:49 -07:00
end
else
buff = buff .. nomsu
end
end
return buff , inline
elseif " String " == _exp_0 then
local buff = " \" "
local longbuff = " \" .. \" \n | "
local inline = true
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
if type ( bit ) == " string " then
bit = bit : gsub ( " \\ " , " \\ \\ " )
buff = buff .. bit : gsub ( " \n " , " \\ n " ) : gsub ( " \" " , " \\ \" " )
longbuff = longbuff .. bit : gsub ( " \n " , " \n | " )
else
local inside , bit_inline = self : tree_to_nomsu ( bit , force_inline )
inline = inline and bit_inline
buff = buff .. " \\ ( " .. tostring ( inside ) .. " ) "
longbuff = longbuff .. " \\ ( " .. tostring ( inside ) .. " ) "
end
end
buff = buff .. " \" "
if force_inline or ( inline and # buff <= 90 ) then
return buff , true
else
return longbuff , false
end
elseif " List " == _exp_0 then
local buff = " [ "
local longbuff = " [..] \n "
local longsep = " "
local longline = 0
local inline = true
for i , bit in ipairs ( tree.value ) do
local bit_inline
nomsu , bit_inline = self : tree_to_nomsu ( bit , force_inline )
inline = inline and bit_inline
if inline then
if i > 1 then
buff = buff .. " , "
end
buff = buff .. nomsu
end
longbuff = longbuff .. ( longsep .. nomsu )
longline = longline + # nomsu
if bit_inline and longline <= 90 then
longsep = " , "
else
longsep = " \n "
end
end
buff = buff .. " ] "
if force_inline or ( inline and # buff <= 90 ) then
return buff , true
else
return longbuff , false
end
elseif " Number " == _exp_0 then
return repr ( tree.value ) , true
elseif " Var " == _exp_0 then
return " % " .. tostring ( tree.value ) , true
elseif " Word " == _exp_0 then
return tree.value , true
else
return self : error ( " Unknown/unimplemented thingy: " .. tostring ( tree.type ) )
end
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
if utils.is_list ( value ) then
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
return " (d{ " .. tostring ( concat ( ( function ( )
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
end ) ( ) , " ; " ) ) .. " }) "
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 ,
2017-12-08 15:37:36 -08:00
tree_to_lua = function ( self , tree , filename )
2017-09-12 21:38:54 -07:00
assert ( tree , " No tree provided. " )
2017-09-20 04:21:46 -07:00
if not tree.type then
2017-10-12 14:39:49 -07:00
self : errorln ( debug.traceback ( ) )
2017-09-21 21:11:13 -07:00
self : 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
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 ]
local expr , statement = self : tree_to_lua ( line , filename )
if statement then
insert ( lua_bits , statement )
end
if expr then
insert ( lua_bits , " ret = " .. tostring ( expr ) .. " ; " )
end
end
return nil , concat ( lua_bits , " \n " )
2017-09-24 20:20:43 -07:00
elseif " Nomsu " == _exp_0 then
2017-12-08 15:37:36 -08:00
return " nomsu:parse( " .. tostring ( repr ( tree.value . src ) ) .. " , " .. tostring ( repr ( tree.line_no ) ) .. " ).value[1] " , nil
2017-09-26 15:27:01 -07:00
elseif " Thunk " == _exp_0 then
2017-09-28 17:49:15 -07:00
local lua_bits = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local arg = _list_0 [ _index_0 ]
2017-12-08 15:37:36 -08:00
local expr , statement = self : tree_to_lua ( arg , filename )
2017-09-28 17:49:15 -07:00
if statement then
insert ( lua_bits , statement )
end
if expr then
2017-10-02 19:00:58 -07:00
insert ( lua_bits , " ret = " .. tostring ( expr ) .. " ; " )
2017-09-28 17:49:15 -07:00
end
end
2017-10-13 18:09:46 -07:00
return ( [ [ ( function ( nomsu , vars )
local ret ;
% s
return ret ;
end ) ] ] ) : format ( concat ( lua_bits , " \n " ) )
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 )
2017-12-04 17:35:47 -08:00
local def = self.defs [ tree.stub ]
2017-10-13 14:15:02 -07:00
if def and def.is_macro then
local expr , statement = self : run_macro ( tree )
if def.whiteset then
if expr then
2017-12-04 17:35:47 -08:00
expr = " (nomsu:assert_permission( " .. tostring ( repr ( tree.stub ) ) .. " ) and " .. tostring ( expr ) .. " ) "
2017-10-13 14:15:02 -07:00
end
if statement then
2017-12-04 17:35:47 -08:00
statement = " nomsu:assert_permission( " .. tostring ( repr ( tree.stub ) ) .. " ); \n " .. statement
2017-10-13 14:15:02 -07:00
end
end
2017-12-11 17:53:23 -08:00
remove ( self.compilestack )
2017-10-13 14:15:02 -07:00
return expr , statement
2017-09-24 20:20:43 -07:00
end
local args = {
2017-12-04 17:35:47 -08:00
repr ( tree.stub ) ,
2017-10-13 16:10:47 -07:00
repr ( tree.line_no )
2017-09-24 20:20:43 -07:00
}
2017-12-04 17:35:47 -08:00
local arg_names , escaped_args
if def then
arg_names , escaped_args = def.arg_names , def.escaped_args
else
arg_names , escaped_args = ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local w = _list_0 [ _index_0 ]
if w.type == " Word " then
_accum_0 [ _len_0 ] = w.value
_len_0 = _len_0 + 1
end
end
return _accum_0
end ) ( ) , { }
end
local arg_num = 1
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 arg = _list_0 [ _index_0 ]
if arg.type == ' Word ' then
_continue_0 = true
break
end
2017-12-04 17:35:47 -08:00
if escaped_args [ arg_names [ arg_num ] ] then
arg = {
type = " Nomsu " ,
2017-12-08 15:37:36 -08:00
value = arg ,
line_no = tree.line_no
2017-12-04 17:35:47 -08:00
}
end
2017-12-08 15:37:36 -08:00
local expr , statement = self : tree_to_lua ( arg , filename )
2017-09-24 20:20:43 -07:00
if statement then
self : error ( " Cannot use [[ " .. tostring ( arg.src ) .. " ]] as a function argument, since it's not an expression. " )
2017-09-12 21:38:54 -07:00
end
2017-09-24 20:20:43 -07:00
insert ( args , expr )
2017-12-04 17:35:47 -08:00
arg_num = arg_num + 1
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
2017-12-11 17:53:23 -08:00
remove ( self.compilestack )
2017-09-24 20:20:43 -07:00
return self.__class : comma_separated_items ( " nomsu:call( " , args , " ) " ) , nil
2017-09-12 21:38:54 -07:00
elseif " String " == _exp_0 then
2017-09-28 17:49:15 -07:00
if self.debug then
self : writeln ( ( colored.bright ( " STRING: " ) ) )
self : print_tree ( tree )
end
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
2017-12-08 15:37:36 -08:00
local expr , statement = self : tree_to_lua ( bit , filename )
2017-09-28 17:49:15 -07:00
if self.debug then
self : writeln ( ( colored.bright ( " INTERP: " ) ) )
self : print_tree ( bit )
self : writeln ( tostring ( colored.bright ( " EXPR: " ) ) .. " " .. tostring ( expr ) .. " , " .. tostring ( colored.bright ( " STATEMENT: " ) ) .. " " .. tostring ( statement ) )
end
2017-09-24 20:20:43 -07:00
if statement then
self : error ( " Cannot use [[ " .. tostring ( bit.src ) .. " ]] as a string interpolation value, since it's not an expression. " )
end
2017-10-13 16:16:07 -07:00
insert ( concat_parts , " nomsu:stringify( " .. tostring ( 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
return " '' " , nil
2017-09-28 17:49:15 -07:00
elseif # concat_parts == 1 then
return concat_parts [ 1 ] , nil
else
return " ( " .. tostring ( concat ( concat_parts , " .. " ) ) .. " ) " , nil
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 ]
2017-12-08 15:37:36 -08:00
local expr , statement = self : tree_to_lua ( item , filename )
2017-09-24 20:20:43 -07:00
if statement then
self : error ( " Cannot use [[ " .. tostring ( item.src ) .. " ]] as a list item, since it's not an expression. " )
end
insert ( items , expr )
2017-09-12 21:38:54 -07:00
end
2017-09-24 20:20:43 -07:00
return self.__class : comma_separated_items ( " { " , items , " } " ) , nil
elseif " Number " == _exp_0 then
2017-09-28 17:49:15 -07:00
return repr ( tree.value ) , nil
2017-09-12 21:38:54 -07:00
elseif " Var " == _exp_0 then
2017-10-13 18:14:18 -07:00
if tree.value : match ( " ^[a-zA-Z_][a-zA-Z0-9_]*$ " ) then
return " vars. " .. tostring ( tree.value ) , nil
else
return " vars[ " .. tostring ( repr ( tree.value ) ) .. " ] " , nil
end
2017-09-12 21:38:54 -07:00
else
2017-09-21 04:51:02 -07:00
return self : 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
2017-10-02 19:00:58 -07:00
if " List " == _exp_0 or " File " == _exp_0 or " Thunk " == _exp_0 or " FunctionCall " == _exp_0 or " String " == _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
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 ,
replaced_vars = function ( self , tree , vars )
if type ( tree ) ~= ' table ' then
return tree
end
local _exp_0 = tree.type
if " Var " == _exp_0 then
2017-09-28 17:49:15 -07:00
if vars [ tree.value ] ~= nil then
2017-09-24 20:20:43 -07:00
tree = vars [ tree.value ]
end
2017-10-02 19:00:58 -07:00
elseif " File " == _exp_0 or " Nomsu " == _exp_0 or " Thunk " == _exp_0 or " List " == _exp_0 or " FunctionCall " == _exp_0 or " String " == _exp_0 then
2017-09-26 15:27:01 -07:00
local new_value = self : replaced_vars ( tree.value , vars )
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
elseif nil == _exp_0 then
local new_values = { }
local any_different = false
for k , v in pairs ( tree ) do
2017-09-26 15:27:01 -07:00
new_values [ k ] = self : replaced_vars ( v , vars )
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
2017-09-26 15:27:01 -07:00
self : error ( " Nothing to get stub from " )
2017-09-21 21:11:13 -07:00
end
if type ( x ) == ' string ' then
2017-12-08 15:37:36 -08:00
x = x : gsub ( " \n %s*%.%. " , " " )
x = lpeg.Cs ( ( operator / ( function ( op )
return " " .. tostring ( op ) .. " "
end ) + 1 ) ^ 0 ) : match ( x )
x = x : gsub ( " %s+ " , " " ) : gsub ( " ^%s* " , " " ) : gsub ( " %s*$ " , " " )
2017-12-04 17:35:47 -08:00
local stub = x : gsub ( " %%%S+ " , " %% " ) : gsub ( " \\ " , " " )
2017-09-28 17:49:15 -07:00
local arg_names
2017-09-21 21:11:13 -07:00
do
local _accum_0 = { }
local _len_0 = 1
2017-12-04 17:35:47 -08:00
for arg in x : gmatch ( " %%([^%s]*) " ) do
2017-09-21 21:11:13 -07:00
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
2017-09-28 17:49:15 -07:00
arg_names = _accum_0
2017-09-21 21:11:13 -07:00
end
2017-12-04 17:35:47 -08:00
local escaped_args = utils.set ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for arg in x : gmatch ( " \\ %%([^%s]*) " ) do
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) )
return stub , arg_names , escaped_args
end
if type ( x ) ~= ' table ' then
self : 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
if " String " == _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
2017-11-01 16:49:11 -07:00
return self : 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
2017-09-24 20:20:43 -07: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-10-12 14:39:49 -07:00
error = function ( self , msg )
2017-12-04 17:35:47 -08:00
local error_msg = colored.red ( " ERROR! " )
2017-10-12 14:39:49 -07:00
if msg then
2017-12-04 17:35:47 -08:00
error_msg = error_msg .. ( " \n " .. ( colored.bright ( colored.yellow ( colored.onred ( msg ) ) ) ) )
2017-09-26 15:27:01 -07:00
end
2017-12-04 17:35:47 -08:00
error_msg = error_msg .. " \n Callstack: "
2017-10-13 16:10:47 -07:00
local maxlen = utils.max ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.callstack
for _index_0 = 1 , # _list_0 do
local c = _list_0 [ _index_0 ]
2017-10-20 15:10:04 -07:00
if c ~= " #macro " then
_accum_0 [ _len_0 ] = # c [ 2 ]
_len_0 = _len_0 + 1
end
2017-10-13 16:10:47 -07:00
end
return _accum_0
end ) ( ) )
2017-09-12 21:38:54 -07:00
for i = # self.callstack , 1 , - 1 do
2017-10-13 16:10:47 -07:00
if self.callstack [ i ] ~= " #macro " then
2017-12-08 15:37:36 -08:00
local line_no = self.callstack [ i ] [ 2 ]
if line_no then
local nums
do
local _accum_0 = { }
local _len_0 = 1
for n in line_no : gmatch ( " :([0-9]+) " ) do
_accum_0 [ _len_0 ] = tonumber ( n )
_len_0 = _len_0 + 1
end
nums = _accum_0
end
line_no = line_no : gsub ( " :.*$ " , " : " .. tostring ( utils.sum ( nums ) - # nums + 1 ) )
end
error_msg = error_msg .. " \n " .. tostring ( ( " %- " .. tostring ( maxlen ) .. " s " ) : format ( line_no ) ) .. " | " .. tostring ( self.callstack [ i ] [ 1 ] )
2017-10-13 16:10:47 -07:00
end
2017-09-12 21:38:54 -07:00
end
2017-12-04 17:35:47 -08:00
error_msg = error_msg .. " \n <top level> "
2017-09-14 18:18:42 -07:00
self.callstack = { }
2017-12-04 17:35:47 -08:00
return error ( error_msg , 3 )
2017-09-12 21:38:54 -07:00
end ,
2017-09-28 17:49:15 -07:00
typecheck = function ( self , vars , varname , desired_type )
local x = vars [ varname ]
if type ( x ) == desired_type then
return x
end
if type ( x ) == ' table ' and x.type == desired_type then
return x
end
2017-12-04 17:35:47 -08:00
return self : error ( " Invalid type for % " .. tostring ( varname ) .. " . Expected " .. tostring ( desired_type ) .. " , but got " .. tostring ( repr ( x ) ) .. " . " )
2017-09-28 17:49:15 -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 )
2017-10-19 18:16:48 -07:00
local nomsu_string_as_lua
2017-12-08 15:37:36 -08:00
nomsu_string_as_lua = function ( self , 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
2017-12-08 15:37:36 -08:00
local expr , statement = self : tree_to_lua ( bit , filename )
2017-10-19 18:16:48 -07:00
if statement then
self : error ( " Cannot use [[ " .. tostring ( bit.src ) .. " ]] as a string interpolation value, since it's not an expression. " )
end
insert ( concat_parts , expr )
end
end
return concat ( concat_parts )
end
2017-09-26 15:27:01 -07:00
local lua_code
lua_code = function ( self , vars )
2017-10-19 18:16:48 -07:00
local lua = nomsu_string_as_lua ( self , vars.code )
2017-09-26 15:27:01 -07:00
return nil , lua
end
2017-10-19 18:16:48 -07:00
self : defmacro ( " lua> %code " , lua_code )
2017-09-26 15:27:01 -07:00
local lua_value
lua_value = function ( self , vars )
2017-10-19 18:16:48 -07:00
local lua = nomsu_string_as_lua ( self , vars.code )
2017-09-26 15:27:01 -07:00
return lua , nil
end
2017-10-19 18:16:48 -07:00
self : defmacro ( " =lua %code " , lua_value )
2017-12-11 17:53:23 -08:00
self : defmacro ( " __src__ %level " , function ( self , vars )
return self : repr ( self : source_code ( self : tree_to_value ( vars.level ) ) )
2017-12-09 16:00:09 -08:00
end )
2017-10-08 18:23:48 -07:00
local run_file
run_file = function ( self , vars )
if vars.filename : match ( " .*%.lua " ) then
return dofile ( vars.filename ) ( self , vars )
end
if vars.filename : match ( " .*%.nom " ) then
if not self.skip_precompiled then
2017-12-14 16:41:22 -08:00
local file = io.open ( vars.filename : gsub ( " %.nom " , " .compiled.nom " ) , " r " )
2017-10-08 18:23:48 -07:00
end
2017-12-14 16:41:22 -08:00
local file = file or io.open ( vars.filename )
2017-09-20 04:21:46 -07:00
if not file then
self : error ( " File does not exist: " .. tostring ( vars.filename ) )
end
2017-10-08 18:23:48 -07:00
local contents = file : read ( ' *a ' )
file : close ( )
return self : run ( contents , vars.filename )
else
return self : error ( " Invalid filetype for " .. tostring ( vars.filename ) )
2017-09-19 00:35:37 -07:00
end
2017-09-26 15:27:01 -07:00
end
2017-10-08 18:23:48 -07:00
self : def ( " run file %filename " , run_file )
local _require
_require = function ( self , vars )
2017-12-04 17:35:47 -08:00
local loaded = self.defs [ " #loaded_files " ]
if not loaded [ vars.filename ] then
loaded [ vars.filename ] = run_file ( self , {
2017-10-08 18:23:48 -07:00
filename = vars.filename
} ) or true
2017-09-20 04:21:46 -07:00
end
2017-12-04 17:35:47 -08:00
return loaded [ vars.filename ]
2017-09-26 15:27:01 -07:00
end
2017-10-08 18:23:48 -07:00
return self : def ( " require %filename " , _require )
2017-09-12 21:38:54 -07:00
end
}
_base_0.__index = _base_0
_class_0 = setmetatable ( {
__init = function ( self , parent )
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
2017-12-04 17:35:47 -08:00
self.defs = {
[ " #vars " ] = { } ,
[ " #loaded_files " ] = { }
}
if parent then
setmetatable ( self.defs , {
__index = parent.defs
} )
setmetatable ( self.defs [ " #vars " ] , {
__index = parent [ " #vars " ]
} )
setmetatable ( self.defs [ " #loaded_files " ] , {
__index = parent [ " #loaded_files " ]
} )
end
2017-09-12 21:38:54 -07:00
self.callstack = { }
2017-12-11 17:53:23 -08:00
self.compilestack = { }
2017-09-12 21:38:54 -07:00
self.debug = false
2017-09-18 12:34:10 -07:00
self.utils = utils
2017-09-21 21:11:13 -07:00
self.repr = function ( self , ... )
return repr ( ... )
end
2017-10-13 16:16:07 -07:00
self.stringify = function ( self , ... )
return utils.stringify ( ... )
end
2017-10-08 15:06:05 -07:00
if not parent then
return self : initialize_core ( )
end
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
2017-09-21 13:30:59 -07:00
self.unescape_string = function ( self , str )
return str : gsub ( " \\ (.) " , ( function ( c )
return STRING_ESCAPES [ c ] or c
end ) )
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
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 )
2017-12-14 14:08:07 -08:00
flag <- " -c " / " -i " / " -p " / " -O " / " --help " / " -h "
2017-10-08 18:23:48 -07:00
input <- " - " / [ ^ ; ] +
output <- " - " / [ ^ ; ] +
] ] , {
set = utils.set
} )
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
2017-10-08 18:23:48 -07:00
local c = NomsuCompiler ( )
2017-12-14 14:08:07 -08:00
c.skip_precompiled = not args.flags [ " -O " ]
2017-10-08 18:23:48 -07:00
if args.input then
if args.flags [ " -c " ] and not args.output then
2017-12-14 16:41:22 -08:00
args.output = args.input : gsub ( " %.nom " , " .compiled.nom " )
2017-10-08 18:23:48 -07:00
end
local compiled_output = nil
if args.flags [ " -p " ] then
local _write = c.write
c.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 ) ( c , { } )
2017-09-12 21:38:54 -07:00
else
2017-10-08 18:23:48 -07:00
local input
if args.input == ' - ' then
input = io.read ( ' *a ' )
else
input = io.open ( args.input ) : read ( " *a " )
end
2017-12-14 16:41:22 -08:00
local vars = { }
local retval , code = c : run ( input , args.input , vars , nil , compiled_output )
2017-10-08 18:23:48 -07:00
end
if args.flags [ " -p " ] then
c.write = _write
2017-09-12 21:38:54 -07:00
end
end
2017-10-20 15:17:57 -07:00
if args.flags [ " -i " ] then
2017-10-31 18:35:40 -07:00
local vars = { }
2017-10-09 04:31:41 -07:00
c : run ( ' require "lib/core.nom" ' , " stdin " )
2017-09-14 15:35:06 -07:00
while true do
2017-10-08 18:23:48 -07:00
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
2017-09-14 15:35:06 -07:00
break
end
2017-10-08 18:23:48 -07:00
local ok , ret = pcall ( function ( )
2017-10-31 18:35:40 -07:00
return c : run ( buff , " stdin " , vars )
2017-10-08 18:23:48 -07:00
end )
if ok and ret ~= nil then
print ( " = " .. repr ( ret ) )
end
2017-09-14 18:18:42 -07:00
end
2017-09-14 15:35:06 -07:00
end
2017-09-12 21:38:54 -07:00
end
2017-09-13 16:22:04 -07:00
return NomsuCompiler