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
local insert , remove , concat
do
local _obj_0 = table
insert , remove , concat = _obj_0.insert , _obj_0.remove , _obj_0.concat
end
local pcall
pcall = function ( fn , ... )
return true , fn ( ... )
end
2017-09-12 21:38:54 -07:00
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-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
}
local check_indent
check_indent = function ( subject , end_pos , spaces )
if # spaces > indent_stack [ # indent_stack ] then
insert ( indent_stack , # spaces )
return end_pos
end
end
local check_dedent
check_dedent = function ( subject , end_pos , spaces )
if # spaces < indent_stack [ # indent_stack ] then
remove ( indent_stack )
return end_pos
end
end
local check_nodent
check_nodent = function ( subject , end_pos , spaces )
if # spaces == indent_stack [ # indent_stack ] then
return end_pos
end
end
2017-09-24 20:20:43 -07:00
local nomsu = [ = [ file <- ( { { | shebang ?
( ignored_line % nl ) *
statements ( nodent statements ) *
( % nl ignored_line ) * % nl ?
( ( { . + } ( " " -> " Unexpected end of file " ) ) => 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-09-24 20:20:43 -07:00
inline_block <- ( { { | " ( " inline_statements " ) " | } } ) -> Block
eol_block <- ( { { | " : " % ws ? noeol_statements eol | } } ) -> Block
indented_block <- ( { { | ( " : " / " (..) " ) indent
statements
( dedent / ( ( { . + } ( " " -> " Error while parsing block " ) ) => error ) )
| } } ) -> Block
2017-09-22 00:03:32 -07:00
2017-09-24 20:20:43 -07:00
inline_nomsu <- ( { ( " \" inline_block ) }) -> Nomsu
eol_nomsu <- ( { ( " \" eol_block ) }) -> Nomsu
indented_nomsu <- ( { ( " \" {indented_block} ) }) -> Nomsu
2017-09-22 00:03:32 -07:00
2017-09-24 20:20:43 -07:00
inline_expression <- number / variable / inline_string / inline_list / inline_block / inline_nomsu
noeol_expression <- indented_string / indented_block / indented_nomsu / indented_list / inline_expression
expression <- eol_block / eol_nomsu / noeol_expression
2017-09-22 00:03:32 -07:00
-- Function calls need at least one word in them
2017-09-24 20:20:43 -07:00
inline_functioncall <- ( { { |
( inline_expression tok_gap ) * word ( tok_gap ( inline_expression / word ) ) *
| } } ) -> FunctionCall
noeol_functioncall <- ( { { |
( noeol_expression tok_gap ) * word ( tok_gap ( noeol_expression / word ) ) *
| } } ) -> FunctionCall
2017-09-22 00:03:32 -07:00
functioncall <- ( { { |
( expression ( dotdot / tok_gap ) ) * word ( ( dotdot / tok_gap ) ( expression / word ) ) *
| } } ) -> FunctionCall
word <- ( { ! number { % wordchar ( ! " ' " % wordchar ) * } } ) -> Word
2017-09-24 20:20:43 -07:00
inline_string <- ( { ' " ' { |
( { ~ ( ( " \\ " -> " \" ) / (' \" ' -> ' " ' ) / (!string_interpolation [^%nl"]))+ ~}
/ string_interpolation ) * | } ' " ' } ) -> String
indented_string <- ( { ' ".." ' indent { |
indented_string_line ( nodent { ~ " " -> "
" ~} indented_string_line)*
| } ( dedent / ( ( { . + } ( " " -> " Error while parsing String " ) ) => error ) )
} ) -> String
indented_string_line <- " | " ( { ~ ( ( " \\ " -> " \" ) / (!string_interpolation [^%nl]))+ ~} / string_interpolation)*
string_interpolation <- " \" (inline_block / indented_block / 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-09-24 20:20:43 -07:00
-- Variables can be nameless (i.e. just %) and can't contain apostrophes
-- which is a hack to allow %foo's to parse as "%foo" and "'s" separately
variable <- ( { ( " % " { ( ! " ' " % wordchar ) * } ) } ) -> 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
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 ) ) )
2017-09-24 20:20:43 -07:00
tok_gap <- % ws / % prev_edge / & ( " [ " / " \" / [.,:;{( " #% ' ])
comma <- % ws ? " , " % ws ?
semicolon <- % ws ? " ; " % ws ?
2017-09-22 00:03:32 -07:00
dotdot <- nodent " .. " % ws ?
] = ]
local whitespace = S ( " \t " ) ^ 1
local defs = {
ws = whitespace ,
nl = P ( " \n " ) ,
2017-09-24 20:20:43 -07:00
tonumber = tonumber ,
2017-09-22 00:03:32 -07:00
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 ) ,
2017-09-24 20:20:43 -07:00
prev_edge = B ( S ( " \t \n .,:;}]) \" \\ " ) ) ,
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 ) .. " ^ "
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 )
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-09-24 20:20:43 -07:00
def = function ( self , invocation , thunk , src )
if type ( invocation ) ~= ' string ' then
self : error ( " Invocation should be string, not: " .. tostring ( repr ( invocation ) ) )
2017-09-12 21:38:54 -07:00
end
if self.debug then
2017-09-24 20:20:43 -07:00
self : writeln ( " Defining rule: " .. tostring ( repr ( invocation ) ) )
2017-09-12 21:38:54 -07:00
end
2017-09-24 20:20:43 -07:00
local stub = invocation : gsub ( " ' " , " ' " ) : gsub ( " %%%S+ " , " %% " ) : gsub ( " %s+ " , " " )
local args
do
local _accum_0 = { }
local _len_0 = 1
for arg in invocation : gmatch ( " %%(%S[^%s']*) " ) do
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
2017-09-18 22:41:50 -07:00
end
2017-09-24 20:20:43 -07:00
args = _accum_0
2017-09-18 22:41:50 -07:00
end
2017-09-24 20:20:43 -07:00
for i = 1 , # args - 1 do
for j = i + 1 , # args do
if args [ i ] == args [ j ] then
self : error ( " Duplicate argument in function def: " .. tostring ( args [ i ] ) )
end
end
2017-09-18 22:41:50 -07:00
end
2017-09-24 20:20:43 -07:00
do
local _with_0 = {
thunk = thunk ,
invocation = invocation ,
args = args ,
src = src ,
is_macro = false
}
self.defs [ invocation ] = _with_0
local _ = nil
return _with_0
2017-09-18 22:41:50 -07:00
end
2017-09-21 21:11:13 -07:00
end ,
2017-09-24 20:20:43 -07:00
defmacro = function ( self , invocation , thunk , src )
do
local _with_0 = self : def ( invocation , thunk , src )
_with_0.is_macro = true
return _with_0
2017-09-12 21:38:54 -07:00
end
2017-09-21 21:11:13 -07:00
end ,
call = function ( self , alias , ... )
2017-09-24 20:20:43 -07:00
local def = self.defs [ alias ]
if def == nil then
2017-09-21 21:11:13 -07:00
self : error ( " Attempt to call undefined function: " .. tostring ( alias ) )
end
2017-09-24 20:20:43 -07:00
if def.is_macro and self.callstack [ # self.callstack ] ~= " #macro " then
2017-09-21 21:11:13 -07:00
self : error ( " Attempt to call macro at runtime: " .. tostring ( alias ) .. " \n This can be caused by using a macro in a function that is defined before the macro. " )
end
2017-09-24 20:20:43 -07:00
if not ( self : check_permission ( def ) ) then
2017-09-21 21:11:13 -07:00
self : error ( " You do not have the authority to call: " .. tostring ( alias ) )
end
2017-09-24 20:20:43 -07:00
local thunk , args
thunk , args = def.thunk , def.args
2017-09-21 21:11:13 -07:00
do
local _tbl_0 = { }
2017-09-24 20:20:43 -07:00
for i , name in ipairs ( args ) 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-22 00:51:53 -07:00
self : writeln ( " Calling " .. tostring ( repr ( alias ) ) .. " with args: " .. tostring ( repr ( args ) ) )
2017-09-21 21:11:13 -07:00
end
2017-09-22 00:27:27 -07:00
insert ( self.callstack , alias )
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
}
remove ( self.callstack )
return unpack ( rets )
end ,
run_macro = function ( self , tree , kind )
if kind == nil then
kind = " Expression "
end
2017-09-24 20:20:43 -07:00
local args , alias
alias , args = self : get_alias ( tree )
insert ( self.callstack , " #macro " )
local expr , statement = self : call ( alias , 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-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-09-22 00:27:27 -07:00
if whiteset [ caller ] 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-14 21:03:42 -07:00
self : writeln ( " 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-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-09-22 00:03:32 -07:00
local tree = nomsu : match ( str )
indent_stack = old_indent_stack
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-09-24 20:20:43 -07:00
run = function ( self , src , filename )
local tree = self : parse ( src , filename )
assert ( tree , " Tree failed to compile: " .. tostring ( src ) )
assert ( tree.type == " File " )
local buffer = { }
local vars = { }
local return_value = nil
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local statement = _list_0 [ _index_0 ]
local ok , expr , statements = pcall ( self.tree_to_lua , self , statement )
if not ok then
self : writeln ( " Error occurred in statement: \n " .. tostring ( statement.src ) )
self : error ( expr )
end
local code_for_statement = ( [ [ return ( function ( nomsu , vars )
% s
return % s
end ) ] ] ) : format ( statements or " " , expr or " " )
if self.debug then
self : writeln ( " RUNNING LUA: \n " .. tostring ( code_for_statement ) )
end
local lua_thunk , err = load ( code_for_statement )
if not lua_thunk then
error ( " Failed to compile generated code: \n " .. tostring ( code_for_statement ) .. " \n \n " .. tostring ( err ) .. " \n \n Produced by statement: \n " .. tostring ( statement.src ) )
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
self : writeln ( " Error occurred in statement: \n " .. tostring ( statement.src ) )
self : error ( return_value )
end
insert ( buffer , tostring ( statements or ' ' ) .. " \n " .. tostring ( expr and " ret = " .. tostring ( expr ) or ' ' ) )
end
local lua_code = ( [ [ return function ( nomsu , vars )
local ret
% s
return ret
end ] ] ) : format ( concat ( buffer , " \n " ) )
return return_value , lua_code
end ,
2017-09-14 02:41:10 -07:00
tree_to_value = function ( self , tree , vars )
2017-09-22 11:56:46 -07:00
local code = " \n return (function(nomsu, 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 ,
2017-09-21 04:04:08 -07:00
tree_to_lua = function ( self , tree )
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-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-09-24 20:20:43 -07:00
return error ( " Should not be converting File to lua through this function. " )
elseif " Nomsu " == _exp_0 then
return repr ( tree.value ) , nil
2017-09-12 21:38:54 -07:00
elseif " Block " == _exp_0 then
2017-09-24 20:20:43 -07:00
local lua_bits = { }
2017-09-12 21:38:54 -07:00
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
2017-09-24 20:20:43 -07:00
local arg = _list_0 [ _index_0 ]
local expr , statement = self : tree_to_lua ( arg )
if expr and not statement and # tree.value == 1 then
return expr , nil
end
if statement then
insert ( lua_bits , statement )
end
if expr then
insert ( lua_bits , " ret = " .. tostring ( expr ) )
2017-09-22 11:44:07 -07:00
end
end
2017-09-24 20:20:43 -07:00
return ( [ [ function ( nomsu , vars )
2017-09-18 12:34:10 -07:00
local ret
2017-09-22 11:44:07 -07:00
% s
return ret
2017-09-24 20:20:43 -07:00
end ] ] ) : format ( concat ( lua_bits , " \n " ) )
2017-09-12 21:38:54 -07:00
elseif " FunctionCall " == _exp_0 then
2017-09-21 21:11:13 -07:00
local alias = self : get_alias ( tree )
if self.defs [ alias ] and self.defs [ alias ] . is_macro then
2017-09-21 04:51:02 -07:00
return self : run_macro ( tree , " Expression " )
2017-09-24 20:20:43 -07:00
end
local args = {
repr ( alias )
}
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
local expr , statement = self : tree_to_lua ( arg )
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 )
_continue_0 = true
until true
if not _continue_0 then
break
2017-09-12 21:38:54 -07:00
end
end
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-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-09-24 20:20:43 -07:00
local expr , statement = self : tree_to_lua ( bit )
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 , " nomsu.utils.repr_if_not_string( " .. tostring ( expr ) .. " ) " )
_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-24 20:20:43 -07:00
return " ( " .. tostring ( concat ( concat_parts , " .. " ) ) .. " ) " , nil
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 ]
local expr , statement = self : tree_to_lua ( item )
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
return repr ( tree.value )
2017-09-12 21:38:54 -07:00
elseif " Var " == _exp_0 then
2017-09-21 21:11:13 -07:00
return " vars[ " .. tostring ( repr ( tree.value ) ) .. " ] "
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-24 20:20:43 -07:00
print_tree = function ( self , tree , ind )
if ind == nil then
ind = " "
end
if type ( tree ) ~= ' table ' or not tree.type then
self : writeln ( tostring ( ind ) .. tostring ( repr ( tree ) ) )
return
end
self : writeln ( tostring ( ind ) .. tostring ( tree.type ) .. " : " )
local _exp_0 = tree.type
if " List " == _exp_0 or " File " == _exp_0 or " Block " == _exp_0 or " FunctionCall " == _exp_0 or " String " == _exp_0 then
local _list_0 = tree.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
self : print_tree ( v , ind .. " " )
end
else
return self : print_tree ( tree.value , ind .. " " )
end
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
if vars [ tree.value ] then
tree = vars [ tree.value ]
end
elseif " File " == _exp_0 or " Thunk " == _exp_0 or " Statement " == _exp_0 or " Block " == _exp_0 or " List " == _exp_0 or " FunctionCall " == _exp_0 or " String " == _exp_0 then
local new_value = self : replaced_vars ( tree.value )
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
new_values [ k ] = self : replaced_vars ( v )
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-21 21:11:13 -07:00
get_alias = function ( self , x )
if not x then
self : error ( " Nothing to get alias from " )
end
if type ( x ) == ' string ' then
local alias = x : gsub ( " ' " , " ' " ) : gsub ( " %%%S+ " , " %% " ) : gsub ( " %s+ " , " " )
local args
do
local _accum_0 = { }
local _len_0 = 1
for arg in x : gmatch ( " %%(%S[^%s']*) " ) do
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
args = _accum_0
end
return alias , args
end
local _exp_0 = x.type
if " String " == _exp_0 then
return self : get_alias ( x.value )
elseif " Statement " == _exp_0 then
return self : get_alias ( x.value )
elseif " FunctionCall " == _exp_0 then
local alias , args = { } , { } , { }
local _list_0 = x.value
for _index_0 = 1 , # _list_0 do
local token = _list_0 [ _index_0 ]
local _exp_1 = token.type
if " Word " == _exp_1 then
insert ( alias , token.value )
elseif " Var " == _exp_1 then
insert ( alias , " % " )
insert ( args , token.value )
2017-09-12 21:38:54 -07:00
else
2017-09-21 21:11:13 -07:00
insert ( alias , " % " )
insert ( args , token )
2017-09-12 21:38:54 -07:00
end
2017-09-21 21:11:13 -07:00
end
return concat ( alias , " " ) , args
end
end ,
get_aliases = function ( self , x )
if not x then
self : error ( " Nothing to get aliases from " )
end
if type ( x ) == ' string ' then
local alias , args = self : get_alias ( x )
return {
[ alias ] = args
}
end
local _exp_0 = x.type
if " String " == _exp_0 then
return self : get_aliases ( {
x.value
} )
elseif " Statement " == _exp_0 then
return self : get_aliases ( {
x.value
} )
elseif " FunctionCall " == _exp_0 then
return self : get_aliases ( {
x
} )
elseif " List " == _exp_0 then
x = x.value
elseif " Block " == _exp_0 then
x = x.value
end
do
local _with_0 = { }
for _index_0 = 1 , # x do
local y = x [ _index_0 ]
local alias , args = self : get_alias ( y )
_with_0 [ alias ] = args
end
return _with_0
2017-09-12 21:38:54 -07:00
end
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-09-12 21:38:54 -07:00
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 ,
initialize_core = function ( self )
2017-09-24 20:20:43 -07:00
self : defmacro ( " lua code %statements with value %value " , function ( self , vars )
2017-09-18 22:41:50 -07:00
local inner_vars = setmetatable ( { } , {
__index = function ( _ , key )
2017-09-21 21:15:37 -07:00
return " vars[ " .. tostring ( repr ( key ) ) .. " ] "
2017-09-18 22:41:50 -07:00
end
} )
2017-09-24 20:20:43 -07:00
local statements = self : tree_to_value ( vars.statements , inner_vars )
local value = self : tree_to_value ( vars.value , inner_vars )
return value , statements
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
2017-09-21 14:11:34 -07:00
self.loaded_files [ vars.filename ] = ( self : run ( file : read ( ' *a ' ) , vars.filename ) ) or true
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-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-09-19 00:35:37 -07:00
self.loaded_files = { }
2017-09-24 20:20:43 -07: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
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-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-24 20:20:43 -07:00
c.debug = true
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-21 04:51:02 -07:00
local retval , code = c : run ( 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-24 20:20:43 -07:00
output : write ( ( [ [ local NomsuCompiler = require ( ' nomsu ' )
2017-09-13 16:22:04 -07:00
local c = NomsuCompiler ( )
2017-09-24 20:20:43 -07:00
local run = % s
return run ( c , { } )
] ] ) : format ( code ) )
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
2017-09-21 21:11:13 -07:00
print ( " = " .. repr ( ret ) )
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