2017-09-12 21:38:54 -07:00
local re = require ( ' re ' )
local lpeg = require ( ' lpeg ' )
local utils = require ( ' utils ' )
local INDENT = " "
lpeg.setmaxstack ( 10000 )
local P , V , S , Cg , C , Cp , B , Cmt
P , V , S , Cg , C , Cp , B , Cmt = lpeg.P , lpeg.V , lpeg.S , lpeg.Cg , lpeg.C , lpeg.Cp , lpeg.B , lpeg.Cmt
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-09-12 21:38:54 -07:00
call = function ( self , fn_name , ... )
local fn_info = self.defs [ fn_name ]
if fn_info == nil then
self : error ( " Attempt to call undefined function: " .. tostring ( fn_name ) )
end
if fn_info.is_macro then
2017-09-18 17:08:35 -07:00
self : error ( " Attempt to call macro at runtime: " .. tostring ( fn_name ) .. " \n This can be caused by using a macro in a function that is defined before the macro. " )
2017-09-12 21:38:54 -07:00
end
if not ( self : check_permission ( fn_name ) ) then
self : error ( " You do not have the authority to call: " .. tostring ( fn_name ) )
end
table.insert ( self.callstack , fn_name )
local fn , arg_names
fn , arg_names = fn_info.fn , fn_info.arg_names
local args
do
local _tbl_0 = { }
2017-09-14 05:44:55 -07:00
for i , name in ipairs ( arg_names [ fn_name ] ) do
2017-09-12 21:38:54 -07:00
_tbl_0 [ name ] = select ( i , ... )
end
args = _tbl_0
end
if self.debug then
2017-09-14 21:03:42 -07:00
self : writeln ( " Calling " .. tostring ( fn_name ) .. " with args: " .. tostring ( utils.repr ( args ) ) )
2017-09-12 21:38:54 -07:00
end
local ret = fn ( self , args )
table.remove ( self.callstack )
return ret
end ,
check_permission = function ( self , fn_name )
local fn_info = self.defs [ fn_name ]
if fn_info == nil then
self : error ( " Undefined function: " .. tostring ( fn_name ) )
end
if fn_info.whiteset == nil then
return true
end
local _list_0 = self.callstack
for _index_0 = 1 , # _list_0 do
local caller = _list_0 [ _index_0 ]
if fn_info.whiteset [ caller ] then
return true
end
end
return false
end ,
2017-09-18 17:08:35 -07:00
def = function ( self , spec , fn , src )
2017-09-12 21:38:54 -07:00
if self.debug then
2017-09-14 21:03:42 -07:00
self : writeln ( " Defining rule: " .. tostring ( spec ) )
2017-09-12 21:38:54 -07:00
end
local invocations , arg_names = self : get_invocations ( spec )
local fn_info = {
fn = fn ,
arg_names = arg_names ,
invocations = invocations ,
2017-09-18 17:08:35 -07:00
src = src ,
2017-09-12 21:38:54 -07:00
is_macro = false
}
for _index_0 = 1 , # invocations do
local invocation = invocations [ _index_0 ]
self.defs [ invocation ] = fn_info
end
end ,
2017-09-18 22:41:50 -07:00
get_invocations_from_definition = function ( self , def , vars )
2017-09-20 04:21:46 -07:00
if def.type == " String " then
2017-09-18 22:41:50 -07:00
return self : tree_to_value ( def , vars )
end
2017-09-20 04:21:46 -07:00
if def.type ~= " List " then
error ( " DEF IS: " .. tostring ( utils.repr ( def ) ) )
self : error ( " Trying to get invocations from " .. tostring ( def.type ) .. " , but expected List or String. " )
2017-09-18 22:41:50 -07:00
end
local invocations = { }
2017-09-20 04:21:46 -07:00
local _list_0 = def.value
2017-09-18 22:41:50 -07:00
for _index_0 = 1 , # _list_0 do
2017-09-20 04:21:46 -07:00
local _continue_0 = false
repeat
local item = _list_0 [ _index_0 ]
if item.type == " String " then
table.insert ( invocations , self : tree_to_value ( item , vars ) )
_continue_0 = true
break
end
if item.type ~= " FunctionCall " then
self : error ( " Invalid list item: " .. tostring ( item.type ) .. " , expected FunctionCall or String " )
2017-09-18 22:41:50 -07:00
end
2017-09-20 04:21:46 -07:00
local name_bits = { }
local _list_1 = item.value
for _index_1 = 1 , # _list_1 do
local token = _list_1 [ _index_1 ]
if token.type == " Word " then
table.insert ( name_bits , token.value )
elseif token.type == " Var " then
table.insert ( name_bits , token.src )
else
self : error ( " Unexpected token type in definition: " .. tostring ( token.type ) .. " (expected Word or Var) " )
end
end
table.insert ( invocations , table.concat ( name_bits , " " ) )
_continue_0 = true
until true
if not _continue_0 then
break
2017-09-18 22:41:50 -07:00
end
end
return invocations
end ,
2017-09-12 21:38:54 -07:00
get_invocations = function ( self , text )
2017-09-18 22:41:50 -07:00
if not text then
self : error ( " No text provided! " )
end
if type ( text ) == ' function ' then
error ( " Function passed to get_invocations " )
end
2017-09-12 21:38:54 -07:00
if type ( text ) == ' string ' then
text = {
text
}
end
local invocations = { }
2017-09-14 05:44:55 -07:00
local arg_names = { }
local prev_arg_names = nil
2017-09-12 21:38:54 -07:00
for _index_0 = 1 , # text do
local _text = text [ _index_0 ]
2017-09-14 05:14:28 -07:00
local invocation = _text : gsub ( " ' " , " ' " ) : gsub ( " %%%S+ " , " %% " ) : gsub ( " %s+ " , " " )
2017-09-12 21:38:54 -07:00
local _arg_names
do
local _accum_0 = { }
local _len_0 = 1
2017-09-14 05:14:28 -07:00
for arg in _text : gmatch ( " %%(%S[^%s']*) " ) do
2017-09-12 21:38:54 -07:00
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
_arg_names = _accum_0
end
table.insert ( invocations , invocation )
2017-09-14 05:44:55 -07:00
if prev_arg_names then
if not utils.equivalent ( utils.set ( prev_arg_names ) , utils.set ( _arg_names ) ) then
2017-09-14 05:49:01 -07:00
self : error ( " Conflicting argument names " .. tostring ( utils.repr ( prev_arg_names ) ) .. " and " .. tostring ( utils.repr ( _arg_names ) ) .. " for " .. tostring ( utils.repr ( text ) ) )
2017-09-12 21:38:54 -07:00
end
else
2017-09-14 05:44:55 -07:00
prev_arg_names = _arg_names
2017-09-12 21:38:54 -07:00
end
2017-09-14 05:44:55 -07:00
arg_names [ invocation ] = _arg_names
2017-09-12 21:38:54 -07:00
end
return invocations , arg_names
end ,
2017-09-18 17:08:35 -07:00
defmacro = function ( self , spec , lua_gen_fn , src )
if self.debug then
self : writeln ( " DEFINING MACRO: " .. tostring ( spec ) .. tostring ( src or " " ) )
end
2017-09-12 21:38:54 -07:00
local invocations , arg_names = self : get_invocations ( spec )
local fn_info = {
fn = lua_gen_fn ,
arg_names = arg_names ,
invocations = invocations ,
2017-09-18 17:08:35 -07:00
src = src ,
2017-09-12 21:38:54 -07:00
is_macro = true
}
for _index_0 = 1 , # invocations do
local invocation = invocations [ _index_0 ]
self.defs [ invocation ] = fn_info
end
end ,
2017-09-20 04:21:46 -07:00
run = function ( self , text , filename )
2017-09-12 21:38:54 -07:00
if self.debug then
2017-09-14 21:03:42 -07:00
self : writeln ( " RUNNING TEXT: \n " .. tostring ( text ) )
2017-09-12 21:38:54 -07:00
end
2017-09-20 04:21:46 -07:00
local code , retval = self : compile ( text , filename )
2017-09-12 21:38:54 -07:00
if self.debug then
2017-09-14 21:03:42 -07:00
self : writeln ( " \n GENERATED LUA CODE: \n " .. tostring ( code ) )
2017-09-18 17:08:35 -07:00
self : writeln ( " \n PRODUCED RETURN VALUE: \n " .. tostring ( retval ) )
2017-09-12 21:38:54 -07:00
end
2017-09-14 18:18:42 -07:00
return retval
2017-09-12 21:38:54 -07:00
end ,
2017-09-18 12:34:10 -07:00
serialize = function ( self , obj )
local _exp_0 = type ( obj )
if " function " == _exp_0 then
2017-09-18 17:08:35 -07:00
error ( " Function serialization is not yet implemented. " )
2017-09-18 12:34:10 -07:00
return " assert(load( " .. utils.repr ( string.dump ( obj ) , true ) .. " )) "
elseif " table " == _exp_0 then
if utils.is_list ( obj ) then
return " { " .. tostring ( table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # obj do
local i = obj [ _index_0 ]
_accum_0 [ _len_0 ] = self : serialize ( i )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " , " ) ) .. " } "
else
return " { " .. tostring ( table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for k , v in pairs ( obj ) do
_accum_0 [ _len_0 ] = " [ " .. tostring ( self : serialize ( k ) ) .. " ]= " .. tostring ( self : serialize ( v ) )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " , " ) ) .. " } "
end
elseif " number " == _exp_0 then
return utils.repr ( obj , true )
elseif " string " == _exp_0 then
return utils.repr ( obj , true )
else
return error ( " Serialization not implemented for: " .. tostring ( type ( obj ) ) )
end
end ,
deserialize = function ( self , str )
local lua_thunk , err = load ( " return (function(compiler,vars) \n return " .. str .. " \n end) " )
if not lua_thunk then
error ( " Failed to compile generated code: \n " .. tostring ( str ) .. " \n \n " .. tostring ( err ) )
end
return ( lua_thunk ( ) ) ( self , { } )
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-14 21:03:42 -07:00
self : writeln ( " PARSING: \n " .. tostring ( str ) )
2017-09-12 21:38:54 -07:00
end
2017-09-20 04:21:46 -07:00
local get_line_indentation
get_line_indentation = function ( line )
local indent_amounts = {
[ " " ] = 1 ,
[ " \t " ] = 4
}
do
local sum = 0
local leading_space = line : match ( " [ \t ]* " )
for c in leading_space : gmatch ( " [ \t ] " ) do
sum = sum + indent_amounts [ c ]
end
return sum
end
end
local indent_stack = {
0
}
local check_indent
check_indent = function ( subject , end_pos , spaces )
local num_spaces = get_line_indentation ( spaces )
if num_spaces > indent_stack [ # indent_stack ] then
table.insert ( indent_stack , num_spaces )
return end_pos
end
end
local check_dedent
check_dedent = function ( subject , end_pos , spaces )
local num_spaces = get_line_indentation ( spaces )
if num_spaces < indent_stack [ # indent_stack ] then
table.remove ( indent_stack )
return end_pos
end
end
local check_nodent
check_nodent = function ( subject , end_pos , spaces )
local num_spaces = get_line_indentation ( spaces )
if num_spaces == indent_stack [ # indent_stack ] then
return end_pos
end
end
local lingo = [ = [ file <- ( { { | shebang ? { : body : block : } % nl * ( ( { . + } ( " " -> " Unexpected end of file " ) ) => error ) ? | } } ) -> File
2017-09-12 21:38:54 -07:00
2017-09-20 04:21:46 -07:00
shebang <- " #! " [ ^% nl ] * % nl
block <- ( { { |
( ignored_line % nl ) *
line_of_statements ( nodent line_of_statements ) *
( % nl ignored_line ) * | } } ) -> Block
inline_block <- ( { { | inline_line_of_statements | } } ) -> Block
line_of_statements <- statement ( % ws ? " ; " % ws ? statement ) *
inline_line_of_statements <- inline_statement ( % ws ? " ; " % ws ? inline_statement ) *
statement <- ( { functioncall / expression } ) -> Statement
inline_statement <- ( { inline_functioncall / expression } ) -> Statement
expression <- (
longstring / string / number / variable / list / thunk / block_functioncall
/ ( " ( " % ws ? ( inline_thunk / inline_functioncall ) % ws ? " ) " ) )
-- Function calls need at least one word in them
functioncall <- ( { { |
( expression ( dotdot / tok_gap ) ) * word ( ( dotdot / tok_gap ) ( expression / word ) ) *
| } } ) -> FunctionCall
inline_functioncall <- ( { { |
( expression tok_gap ) * word ( tok_gap ( expression / word ) ) *
| } } ) -> FunctionCall
block_functioncall <- " (..) " indent functioncall ( dedent / ( ( { . + } ( " " -> " Error while parsing block function call " ) ) => error ) )
2017-09-12 21:38:54 -07:00
2017-09-14 04:33:45 -07:00
word <- ( { ! number { % wordchar ( ! " ' " % wordchar ) * } } ) -> Word
2017-09-20 04:21:46 -07:00
thunk <- ( { " : " ( ( indent block ( dedent / ( ( { . + } ( " " -> " Error while parsing thunk " ) ) => error ) ) )
/ ( % ws ? inline_block ) ) } ) -> Thunk
inline_thunk <- ( { " : " % ws ? inline_block } ) -> Thunk
2017-09-12 21:38:54 -07:00
2017-09-14 00:09:54 -07:00
string <- ( { ( ! longstring ) ' " ' { ( ( " \" [^%nl]) / [^ " % nl ] ) * } ' " ' } ) -> String
2017-09-20 04:21:46 -07:00
2017-09-14 02:41:10 -07:00
longstring <- ( { ' ".." ' % ws ?
2017-09-20 04:21:46 -07:00
{ | ( longstring_line ( indent
longstring_line ( nodent longstring_line ) *
( dedent / longstring_error ) ) ? )
/ ( indent
longstring_line ( nodent longstring_line ) *
( dedent / longstring_error ) ) | } } ) -> Longstring
longstring_line <- " | " { | ( { ( " \\ " / ( ! string_interpolation [ ^% nl ] ) ) + } / string_interpolation ) * | }
longstring_error <- ( ( { . + } ( " " -> " Error while parsing Longstring " ) ) => error )
2017-09-20 05:02:53 -07:00
string_interpolation <- " \" %ws? (((inline_functioncall / expression) dotdot?) / dotdot) %ws? " \ "
2017-09-12 21:38:54 -07:00
2017-09-20 04:21:46 -07:00
number <- ( { { " - " ? ( ( [ 0 - 9 ] + " . " [ 0 - 9 ] + ) / ( " . " [ 0 - 9 ] + ) / ( [ 0 - 9 ] + ) ) } } ) -> Number
-- Hack to allow %foo's to parse as "%foo" and "'s" separately
variable <- ( { ( " % " { % wordchar ( ! " ' " % wordchar ) * } ) } ) -> Var
2017-09-12 21:38:54 -07:00
list <- ( { { |
2017-09-20 04:21:46 -07:00
( " [..] " indent
list_line ( nodent list_line ) *
( dedent / ( ( { . + } ( " " -> " Error while parsing list " ) ) => error ) ) )
/ ( " [ " % ws ? ( list_line % ws ? ) ? " ] " )
2017-09-12 21:38:54 -07:00
| } } ) -> List
2017-09-20 04:21:46 -07:00
list_line <- list_bit ( % ws ? " , " tok_gap list_bit ) * ( % ws ? " , " ) ?
list_bit <- inline_functioncall / expression
block_comment <- " #.. " [ ^% nl ] * indent [ ^% nl ] * ( % nl ( ( % ws ? ( ! . / &% nl ) ) / ( ! % dedented [ ^% nl ] * ) ) ) *
line_comment <- " # " [ ^% nl ] *
eol <- % ws ? line_comment ? ( ! . / &% nl )
ignored_line <- ( % nodented ( block_comment / line_comment ) ) / ( % ws ? ( ! . / &% nl ) )
indent <- eol ( % nl ignored_line ) * % nl % indented
nodent <- eol ( % nl ignored_line ) * % nl % nodented
dedent <- eol ( % nl ignored_line ) * ( ( ( ! . ) &% dedented ) / ( & ( % nl % dedented ) ) )
tok_gap <- % ws / % prev_edge / & ( " [ " / [ . , : ; { ( " #%'])
dotdot <- nodent " .. " % ws ?
2017-09-12 21:38:54 -07:00
] = ]
2017-09-20 04:21:46 -07:00
local whitespace = S ( " \t " ) ^ 1
local defs = {
ws = whitespace ,
nl = P ( " \n " ) ,
wordchar = P ( 1 ) - S ( ' \t \n \r %#:;,.{}[]()" \\ ' ) ,
indented = Cmt ( S ( " \t " ) ^ 0 * ( # ( P ( 1 ) - S ( " \t \n " ) + ( - P ( 1 ) ) ) ) , check_indent ) ,
nodented = Cmt ( S ( " \t " ) ^ 0 * ( # ( P ( 1 ) - S ( " \t \n " ) + ( - P ( 1 ) ) ) ) , check_nodent ) ,
dedented = Cmt ( S ( " \t " ) ^ 0 * ( # ( P ( 1 ) - S ( " \t \n " ) + ( - P ( 1 ) ) ) ) , check_dedent ) ,
prev_edge = B ( S ( " \t \n .,:;}]) \" " ) ) ,
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 ) .. " ^ "
return error ( " \n " .. tostring ( err_msg or " Parse error " ) .. " in " .. tostring ( filename ) .. " on line " .. tostring ( line_no ) .. " : \n \n " .. tostring ( prev_line ) .. " \n " .. tostring ( err_line ) .. " \n " .. tostring ( pointer ) .. " \n " .. tostring ( next_line ) .. " \n " )
end
}
setmetatable ( defs , {
__index = function ( t , key )
local fn
fn = function ( src , value , errors )
local token = {
type = key ,
src = src ,
value = value ,
errors = errors
}
return token
end
t [ key ] = fn
return fn
end
} )
lingo = re.compile ( lingo , defs )
2017-09-12 21:38:54 -07:00
local tree = lingo : match ( str : gsub ( " \r " , " " ) .. " \n " )
if self.debug then
2017-09-14 21:03:42 -07:00
self : writeln ( " \n PARSE TREE: " )
2017-09-12 21:38:54 -07:00
self : print_tree ( tree )
end
assert ( tree , " Failed to parse: " .. tostring ( str ) )
return tree
end ,
2017-09-14 02:41:10 -07:00
tree_to_value = function ( self , tree , vars )
2017-09-18 12:34:10 -07:00
local code = " \n return (function(compiler, vars) \n return " .. tostring ( self : tree_to_lua ( tree ) ) .. " \n end) "
2017-09-12 21:38:54 -07:00
local lua_thunk , err = load ( code )
if not lua_thunk then
error ( " Failed to compile generated code: \n " .. tostring ( code ) .. " \n \n " .. tostring ( err ) )
end
2017-09-14 02:41:10 -07:00
return ( lua_thunk ( ) ) ( self , vars or { } )
2017-09-12 21:38:54 -07:00
end ,
tree_to_lua = function ( self , tree , kind )
if kind == nil then
kind = " Expression "
end
assert ( tree , " No tree provided. " )
2017-09-20 04:21:46 -07:00
if not tree.type then
self : error ( " Invalid tree: " .. tostring ( utils.repr ( tree ) ) )
end
2017-09-12 21:38:54 -07:00
local indent = " "
local buffer = { }
2017-09-14 18:18:42 -07:00
local return_value = nil
2017-09-12 21:38:54 -07:00
local to_lua
to_lua = function ( t , kind )
local ret = self : tree_to_lua ( t , kind )
return ret
end
local add
add = function ( code )
return table.insert ( buffer , code )
end
local _exp_0 = tree.type
if " File " == _exp_0 then
add ( [ [ return ( function ( compiler , vars )
local ret ] ] )
local vars = { }
local _list_0 = tree.value . body.value
for _index_0 = 1 , # _list_0 do
local statement = _list_0 [ _index_0 ]
2017-09-14 18:18:42 -07:00
local code = to_lua ( statement , " Statement " )
2017-09-18 12:34:10 -07:00
local lua_code = " \n return (function(compiler, vars) \n " .. tostring ( code ) .. " \n end) "
2017-09-14 19:45:36 -07:00
local lua_thunk , err = load ( lua_code )
2017-09-12 21:38:54 -07:00
if not lua_thunk then
2017-09-14 02:41:10 -07:00
error ( " Failed to compile generated code: \n " .. tostring ( code ) .. " \n \n " .. tostring ( err ) .. " \n \n Produced by statement: \n " .. tostring ( utils.repr ( statement ) ) )
2017-09-12 21:38:54 -07:00
end
2017-09-14 19:45:36 -07:00
local value = lua_thunk ( )
return_value = value ( self , vars )
2017-09-12 21:38:54 -07:00
add ( code )
end
add ( [ [ return ret
end )
] ] )
elseif " Block " == _exp_0 then
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local statement = _list_0 [ _index_0 ]
add ( to_lua ( statement ) )
end
elseif " Thunk " == _exp_0 then
assert ( tree.value . type == " Block " , " Non-block value in Thunk " )
add ( [ [ ( function ( compiler , vars )
2017-09-18 12:34:10 -07:00
local ret
] ] .. to_lua ( tree.value ) .. " \n " .. [ [ return ret
2017-09-12 21:38:54 -07:00
end )
] ] )
elseif " Statement " == _exp_0 then
if tree.value . type == " FunctionCall " then
local name = self : fn_name_from_tree ( tree.value )
if self.defs [ name ] and self.defs [ name ] . is_macro then
add ( self : run_macro ( tree.value , " Statement " ) )
else
add ( " ret = " .. ( to_lua ( tree.value ) : match ( " %s*(.*) " ) ) )
end
else
add ( " ret = " .. ( to_lua ( tree.value ) : match ( " %s*(.*) " ) ) )
end
elseif " FunctionCall " == _exp_0 then
local name = self : fn_name_from_tree ( tree )
if self.defs [ name ] and self.defs [ name ] . is_macro then
add ( self : run_macro ( tree , " Expression " ) )
else
local args
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local a = _list_0 [ _index_0 ]
if a.type ~= " Word " then
_accum_0 [ _len_0 ] = to_lua ( a )
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
table.insert ( args , 1 , utils.repr ( name , true ) )
add ( self.__class : comma_separated_items ( " compiler:call( " , args , " ) " ) )
end
elseif " String " == _exp_0 then
local escapes = {
n = " \n " ,
t = " \t " ,
b = " \b " ,
a = " \a " ,
v = " \v " ,
f = " \f " ,
r = " \r "
}
local unescaped = tree.value : gsub ( " \\ (.) " , ( function ( c )
return escapes [ c ] or c
end ) )
add ( utils.repr ( unescaped , true ) )
elseif " Longstring " == _exp_0 then
2017-09-14 02:41:10 -07:00
local concat_parts = { }
local string_buffer = " "
for i , line in ipairs ( tree.value ) do
if i > 1 then
string_buffer = string_buffer .. " \n "
2017-09-12 21:38:54 -07:00
end
2017-09-14 02:41:10 -07:00
for _index_0 = 1 , # line do
local bit = line [ _index_0 ]
if type ( bit ) == " string " then
string_buffer = string_buffer .. bit : gsub ( " \\ \\ " , " \\ " )
else
if string_buffer ~= " " then
table.insert ( concat_parts , utils.repr ( string_buffer , true ) )
string_buffer = " "
end
2017-09-18 12:34:10 -07:00
table.insert ( concat_parts , " compiler.utils.repr( " .. tostring ( to_lua ( bit ) ) .. " ) " )
2017-09-14 02:41:10 -07:00
end
end
end
if string_buffer ~= " " then
table.insert ( concat_parts , utils.repr ( string_buffer , true ) )
end
if # concat_parts == 0 then
add ( " '' " )
elseif # concat_parts == 1 then
add ( concat_parts [ 1 ] )
else
add ( " ( " .. tostring ( table.concat ( concat_parts , " .. " ) ) .. " ) " )
2017-09-12 21:38:54 -07:00
end
elseif " Number " == _exp_0 then
add ( tree.value )
elseif " List " == _exp_0 then
if # tree.value == 0 then
add ( " {} " )
elseif # tree.value == 1 then
add ( " { " .. tostring ( to_lua ( tree.value [ 1 ] ) ) .. " } " )
else
add ( self.__class : comma_separated_items ( " { " , ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local item = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = to_lua ( item )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " } " ) )
end
elseif " Var " == _exp_0 then
add ( " vars[ " .. tostring ( utils.repr ( tree.value , true ) ) .. " ] " )
else
2017-09-19 00:30:47 -07:00
self : error ( " Unknown/unimplemented thingy: " .. tostring ( tree.type ) )
2017-09-12 21:38:54 -07:00
end
buffer = table.concat ( buffer , " \n " )
2017-09-14 18:18:42 -07:00
return buffer , return_value
2017-09-12 21:38:54 -07:00
end ,
fn_name_from_tree = function ( self , tree )
assert ( tree.type == " FunctionCall " , " Attempt to get fn name from non-functioncall tree: " .. tostring ( tree.type ) )
local name_bits = { }
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local token = _list_0 [ _index_0 ]
table.insert ( name_bits , ( function ( )
if token.type == " Word " then
return token.value
else
return " % "
end
end ) ( ) )
end
return table.concat ( name_bits , " " )
end ,
run_macro = function ( self , tree , kind )
if kind == nil then
kind = " Expression "
end
local name = self : fn_name_from_tree ( tree )
if not ( self.defs [ name ] and self.defs [ name ] . is_macro ) then
self : error ( " Macro not found: " .. tostring ( name ) )
end
if not ( self : check_permission ( name ) ) then
self : error ( " You do not have the authority to call: " .. tostring ( name ) )
end
local fn , arg_names
do
local _obj_0 = self.defs [ name ]
fn , arg_names = _obj_0.fn , _obj_0.arg_names
end
local args
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local a = _list_0 [ _index_0 ]
if a.type ~= " Word " then
_accum_0 [ _len_0 ] = a
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
do
local _tbl_0 = { }
2017-09-14 05:44:55 -07:00
for i , name in ipairs ( arg_names [ name ] ) do
2017-09-12 21:38:54 -07:00
_tbl_0 [ name ] = args [ i ]
end
args = _tbl_0
end
table.insert ( self.callstack , name )
local ret , manual_mode = fn ( self , args , kind )
table.remove ( self.callstack )
if not ret then
self : error ( " No return value for macro: " .. tostring ( name ) )
end
if kind == " Statement " and not manual_mode then
ret = " ret = " .. ret
end
return ret
end ,
_yield_tree = function ( self , tree , indent_level )
if indent_level == nil then
indent_level = 0
end
local ind
ind = function ( s )
return INDENT : rep ( indent_level ) .. s
end
local _exp_0 = tree.type
if " File " == _exp_0 then
coroutine.yield ( ind ( " File: " ) )
self : _yield_tree ( tree.value . body , indent_level + 1 )
elseif " Errors " == _exp_0 then
coroutine.yield ( ind ( " Error: \n " .. tostring ( tree.value ) ) )
elseif " Block " == _exp_0 then
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local chunk = _list_0 [ _index_0 ]
self : _yield_tree ( chunk , indent_level )
end
elseif " Thunk " == _exp_0 then
coroutine.yield ( ind ( " Thunk: " ) )
self : _yield_tree ( tree.value , indent_level + 1 )
elseif " Statement " == _exp_0 then
self : _yield_tree ( tree.value , indent_level )
elseif " FunctionCall " == _exp_0 then
local name = self : fn_name_from_tree ( tree )
local args
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local a = _list_0 [ _index_0 ]
if a.type ~= " Word " then
_accum_0 [ _len_0 ] = a
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
if # args == 0 then
coroutine.yield ( ind ( " Call [ " .. tostring ( name ) .. " ]! " ) )
else
coroutine.yield ( ind ( " Call [ " .. tostring ( name ) .. " ]: " ) )
for _index_0 = 1 , # args do
local a = args [ _index_0 ]
self : _yield_tree ( a , indent_level + 1 )
end
end
elseif " String " == _exp_0 then
coroutine.yield ( ind ( utils.repr ( tree.value , true ) ) )
elseif " Longstring " == _exp_0 then
coroutine.yield ( ind ( utils.repr ( tree.value , true ) ) )
elseif " Number " == _exp_0 then
coroutine.yield ( ind ( tree.value ) )
elseif " List " == _exp_0 then
if # tree.value == 0 then
coroutine.yield ( ind ( " <Empty List> " ) )
else
coroutine.yield ( ind ( " List: " ) )
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local item = _list_0 [ _index_0 ]
self : _yield_tree ( item , indent_level + 1 )
end
end
elseif " Var " == _exp_0 then
coroutine.yield ( ind ( " Var[ " .. tostring ( utils.repr ( tree.value ) ) .. " ] " ) )
else
error ( " Unknown/unimplemented thingy: " .. tostring ( tree.type ) )
end
return nil
end ,
print_tree = function ( self , tree )
for line in coroutine.wrap ( function ( )
return self : _yield_tree ( tree )
end ) do
2017-09-14 21:03:42 -07:00
self : writeln ( line )
2017-09-12 21:38:54 -07:00
end
end ,
stringify_tree = function ( self , tree )
local result = { }
for line in coroutine.wrap ( function ( )
return self : _yield_tree ( tree )
end ) do
table.insert ( result , line )
end
return table.concat ( result , " \n " )
end ,
2017-09-20 04:21:46 -07:00
compile = function ( self , src , filename , output_file )
2017-09-12 21:38:54 -07:00
if output_file == nil then
output_file = nil
end
if self.debug then
2017-09-14 21:03:42 -07:00
self : writeln ( " COMPILING: \n " .. tostring ( src ) )
2017-09-12 21:38:54 -07:00
end
2017-09-20 04:21:46 -07:00
local tree = self : parse ( src , filename )
2017-09-12 21:38:54 -07:00
assert ( tree , " Tree failed to compile: " .. tostring ( src ) )
2017-09-14 18:18:42 -07:00
local code , retval = self : tree_to_lua ( tree )
2017-09-12 21:38:54 -07:00
if output_file then
local output = io.open ( output_file , " w " )
output : write ( code )
end
2017-09-14 18:18:42 -07:00
return code , retval
2017-09-12 21:38:54 -07:00
end ,
error = function ( self , ... )
2017-09-14 21:03:42 -07:00
self : writeln ( " ERROR! " )
self : writeln ( ... )
self : writeln ( " Callstack: " )
2017-09-12 21:38:54 -07:00
for i = # self.callstack , 1 , - 1 do
2017-09-14 21:03:42 -07:00
self : writeln ( " " .. tostring ( self.callstack [ i ] ) )
2017-09-12 21:38:54 -07:00
end
2017-09-14 21:03:42 -07:00
self : writeln ( " <top level> " )
2017-09-14 18:18:42 -07:00
self.callstack = { }
2017-09-12 21:38:54 -07:00
return error ( )
end ,
2017-09-20 04:21:46 -07:00
test = function ( self , src , filename , expected )
2017-09-12 21:38:54 -07:00
local i = 1
while i ~= nil do
local start , stop = src : find ( " \n \n " , i )
local test = src : sub ( i , start )
i = stop
start , stop = test : find ( " === " )
if not start or not stop then
self : error ( " WHERE'S THE ===? in: \n " .. tostring ( test ) )
end
local test_src
test_src , expected = test : sub ( 1 , start - 1 ) , test : sub ( stop + 1 , - 1 )
expected = expected : match ( ' [ \n ]*(.*[^ \n ]) ' )
2017-09-20 04:21:46 -07:00
local tree = self : parse ( test_src , filename )
2017-09-12 21:38:54 -07:00
local got = self : stringify_tree ( tree.value . body )
if got ~= expected then
self : error ( " TEST FAILED! \n Source: \n " .. tostring ( test_src ) .. " \n Expected: \n " .. tostring ( expected ) .. " \n \n Got: \n " .. tostring ( got ) )
end
end
end ,
initialize_core = function ( self )
local as_lua_code
2017-09-14 02:41:10 -07:00
as_lua_code = function ( self , str , vars )
2017-09-12 21:38:54 -07:00
local _exp_0 = str.type
if " String " == _exp_0 then
2017-09-14 02:41:10 -07:00
return self : tree_to_value ( str , vars )
2017-09-12 21:38:54 -07:00
elseif " Longstring " == _exp_0 then
2017-09-14 02:41:10 -07:00
return self : tree_to_value ( str , vars )
2017-09-12 21:38:54 -07:00
else
return self : tree_to_lua ( str )
end
end
self : defmacro ( [[lua block %lua_code]] , function ( self , vars , kind )
if kind == " Expression " then
error ( " Expected to be in statement. " )
end
2017-09-18 22:41:50 -07:00
local inner_vars = setmetatable ( { } , {
__index = function ( _ , key )
return " vars[ " .. tostring ( utils.repr ( key , true ) ) .. " ] "
end
} )
return " do \n " .. self : tree_to_value ( vars.lua_code , inner_vars ) .. " \n end " , true
2017-09-12 21:38:54 -07:00
end )
self : defmacro ( [[lua expr %lua_code]] , function ( self , vars , kind )
local lua_code = vars.lua_code . value
2017-09-18 22:41:50 -07:00
local inner_vars = setmetatable ( { } , {
__index = function ( _ , key )
return " vars[ " .. tostring ( utils.repr ( key , true ) ) .. " ] "
end
} )
return self : tree_to_value ( vars.lua_code , inner_vars )
2017-09-12 21:38:54 -07:00
end )
2017-09-19 00:35:37 -07:00
self : def ( " require %filename " , function ( self , vars )
if not self.loaded_files [ vars.filename ] then
local file = 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
self.loaded_files [ vars.filename ] = self : run ( file : read ( ' *a ' ) , vars.filename )
2017-09-19 00:35:37 -07:00
end
return self.loaded_files [ vars.filename ]
end )
2017-09-12 21:38:54 -07:00
return self : def ( " run file %filename " , function ( self , vars )
local file = 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
return self : run ( file : read ( ' *a ' ) , vars.filename )
2017-09-12 21:38:54 -07:00
end )
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-09-12 21:38:54 -07:00
self.defs = setmetatable ( { } , {
__index = parent and parent.defs
} )
self.callstack = { }
self.debug = false
2017-09-14 21:03:42 -07:00
self : initialize_core ( )
2017-09-18 12:34:10 -07:00
self.utils = utils
2017-09-19 00:35:37 -07:00
self.loaded_files = { }
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
self.comma_separated_items = function ( self , open , items , close )
return utils.accumulate ( " \n " , function ( )
local buffer = open
local so_far = 0
for i , item in ipairs ( items ) do
if i < # items then
item = item .. " , "
end
if so_far + # item >= 80 and # buffer > 0 then
coroutine.yield ( buffer )
so_far = so_far - # buffer
buffer = item
else
so_far = so_far + # item
buffer = buffer .. item
end
end
buffer = buffer .. close
return coroutine.yield ( buffer )
end )
end
2017-09-13 16:22:04 -07:00
NomsuCompiler = _class_0
2017-09-12 21:38:54 -07:00
end
2017-09-12 23:12:45 -07:00
if arg and arg [ 1 ] then
2017-09-13 16:22:04 -07:00
local c = NomsuCompiler ( )
2017-09-12 21:38:54 -07:00
local input = io.open ( arg [ 1 ] ) : read ( " *a " )
2017-09-14 21:03:42 -07:00
local _write = c.write
2017-09-12 21:38:54 -07:00
if arg [ 2 ] == " - " then
2017-09-14 21:03:42 -07:00
c.write = function ( ) end
2017-09-12 21:38:54 -07:00
end
2017-09-20 04:21:46 -07:00
local code , retval = c : compile ( input , arg [ 1 ] )
2017-09-14 21:03:42 -07:00
c.write = _write
2017-09-12 21:38:54 -07:00
if arg [ 2 ] then
local output
if arg [ 2 ] == " - " then
output = io.output ( )
else
output = io.open ( arg [ 2 ] , ' w ' )
end
2017-09-18 12:34:10 -07:00
output : write ( [ [ local load = function ( )
2017-09-12 21:38:54 -07:00
] ] )
output : write ( code )
output : write ( [ [
end
2017-09-13 16:22:04 -07:00
local NomsuCompiler = require ( ' nomsu ' )
local c = NomsuCompiler ( )
2017-09-14 18:18:42 -07:00
return load ( ) ( c , { } )
2017-09-12 21:38:54 -07:00
] ] )
end
2017-09-14 15:35:06 -07:00
elseif arg then
local c = NomsuCompiler ( )
2017-09-20 04:21:46 -07:00
c : run ( ' require "lib/core.nom" ' )
2017-09-14 15:35:06 -07:00
while true do
local buff = " "
while true do
io.write ( " >> " )
local line = io.read ( " *L " )
if line == " \n " or not line then
break
end
buff = buff .. line
end
if # buff == 0 then
break
end
2017-09-14 18:18:42 -07:00
local ok , ret = pcall ( function ( )
return c : run ( buff )
end )
if ok and ret ~= nil then
print ( " = " .. utils.repr ( ret , true ) )
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