2018-06-24 18:20:11 -07:00
if NOMSU_VERSION and NOMSU_PREFIX then
2018-06-23 17:22:23 -07:00
local ver_bits
do
local _accum_0 = { }
local _len_0 = 1
for ver_bit in NOMSU_VERSION : gmatch ( " [0-9]+ " ) do
_accum_0 [ _len_0 ] = ver_bit
_len_0 = _len_0 + 1
end
ver_bits = _accum_0
end
local partial_vers
do
local _accum_0 = { }
local _len_0 = 1
for i = # ver_bits , 1 , - 1 do
_accum_0 [ _len_0 ] = table.concat ( ver_bits , ' . ' , 1 , i )
_len_0 = _len_0 + 1
end
partial_vers = _accum_0
end
package.path = table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # partial_vers do
local v = partial_vers [ _index_0 ]
2018-06-24 18:20:11 -07:00
_accum_0 [ _len_0 ] = tostring ( NOMSU_PREFIX ) .. " /share/nomsu/ " .. tostring ( v ) .. " /?.lua "
2018-06-23 17:22:23 -07:00
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " ; " ) .. " ; " .. package.path
package.cpath = table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # partial_vers do
local v = partial_vers [ _index_0 ]
2018-06-24 18:20:11 -07:00
_accum_0 [ _len_0 ] = tostring ( NOMSU_PREFIX ) .. " /lib/nomsu/ " .. tostring ( v ) .. " /?.so "
2018-06-23 17:22:23 -07:00
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , " ; " ) .. " ; " .. package.cpath
package.nomsupath = table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # partial_vers do
local v = partial_vers [ _index_0 ]
2018-06-24 18:20:11 -07:00
_accum_0 [ _len_0 ] = tostring ( NOMSU_PREFIX ) .. " /share/nomsu/ " .. tostring ( v )
2018-06-23 17:22:23 -07:00
_len_0 = _len_0 + 1
end
return _accum_0
2018-07-24 15:08:44 -07:00
end ) ( ) , " ; " ) .. " ;. "
else
package.nomsupath = " . "
2018-06-23 17:22:23 -07:00
end
2018-06-21 19:12:59 -07:00
local EXIT_SUCCESS , EXIT_FAILURE = 0 , 1
2018-06-19 01:27:32 -07:00
local usage = [ = [ Nomsu Compiler
2018-04-28 17:08:28 -07:00
2018-07-24 15:25:03 -07:00
Usage : ( nomsu | lua nomsu.lua | moon nomsu.moon ) [ - V version ] [ - O optimization level ] [ - v ] [ - c ] [ - s ] [ - t ] [ - I file ] [ --help | -h] [--version] [--no-core] [file [nomsu args...]]
2018-04-28 17:08:28 -07:00
OPTIONS
2018-07-24 15:25:03 -07:00
- O < level > Run the compiler with the given optimization level ( > 0 : use precompiled . lua versions of Nomsu files , when available ) .
2018-07-09 16:58:46 -07:00
- v Verbose : print compiled lua code .
- c Compile the input files into a . lua files .
2018-07-21 14:43:49 -07:00
- e Execute the specified string .
2018-07-09 16:58:46 -07:00
- s Check the input files for syntax errors .
- I < file > Add an additional input file or directory .
2018-07-10 15:06:02 -07:00
- d < debugger > Attempt to use the specified debugger to wrap the main body of execution .
2018-04-28 17:08:28 -07:00
- h /-- help Print this message .
2018-06-23 00:57:31 -07:00
--version Print the version number and exit.
2018-07-23 14:53:12 -07:00
--no-core Skip loading the Nomsu core by default.
2018-07-09 16:58:46 -07:00
- V specify which Nomsu version is desired .
< file > The Nomsu file to run ( can be " - " to use stdin ) .
2018-06-19 01:27:32 -07:00
] = ]
2018-06-21 19:12:59 -07:00
local ok , _ = pcall ( function ( )
lpeg = require ( ' lpeg ' )
re = require ( ' re ' )
end )
if not ok then
print ( " Error: unable to find the 'lpeg' Lua module. Please install LPEG either from http://www.inf.puc-rio.br/~roberto/lpeg/re.html or, if you use luarocks: `luarocks install lpeg` " )
os.exit ( EXIT_FAILURE )
end
2018-07-21 14:43:49 -07:00
local Files = require ( " files " )
2018-06-19 15:24:24 -07:00
local Errhand = require ( " error_handling " )
2018-06-19 01:27:32 -07:00
local NomsuCompiler = require ( " nomsu_compiler " )
local NomsuCode , LuaCode , Source
do
local _obj_0 = require ( " code_obj " )
NomsuCode , LuaCode , Source = _obj_0.NomsuCode , _obj_0.LuaCode , _obj_0.Source
end
2018-06-23 00:57:31 -07:00
local repr
repr = require ( " utils " ) . repr
2018-06-19 01:27:32 -07:00
if not arg or debug.getinfo ( 2 ) . func == require then
return NomsuCompiler
end
2018-07-09 16:58:46 -07:00
local file_queue = { }
2018-07-21 14:43:49 -07:00
local sep = " \0 "
local parser = re.compile ( [ [ args <- { | ( flag % sep ) * ( ( { ~ file ~ } -> add_file ) % sep ) ? { : nomsu_args : { | ( { ( ! % sep . ) * } % sep ) * | } : } % sep ? | } ! .
2018-06-19 01:27:32 -07:00
flag <-
2018-07-24 15:25:03 -07:00
{ : optimization : " -O " ( % sep ? ( ( [ 0 - 9 ] + ) -> tonumber ) ) ? : }
2018-07-21 14:43:49 -07:00
/ ( " -I " % sep ? ( { ~ file ~ } -> add_file ) )
/ ( " -e " % sep ? ( ( { } { ~ file ~ } ) -> add_exec_string ) )
2018-07-09 16:58:46 -07:00
/ ( { : check_syntax : ( " -s " -> true ) : } )
/ ( { : compile : ( " -c " -> true ) : } )
2018-07-21 14:43:49 -07:00
/ ( { : compile : ( " -c " -> true ) : } )
2018-06-19 01:27:32 -07:00
/ { : verbose : ( " -v " -> true ) : }
/ { : help : ( ( " -h " / " --help " ) -> true ) : }
2018-06-23 00:57:31 -07:00
/ { : version : ( " --version " -> true ) : }
2018-07-23 14:53:12 -07:00
/ { : no_core : ( " --no-core " -> true ) : }
2018-07-21 14:43:49 -07:00
/ { : debugger : ( " -d " % sep ? { ( ! % sep . ) * } ) : }
/ { : requested_version : " -V " ( % sep ? { ( [ 0 - 9. ] ) + } ) ? : }
file <- ( " - " -> " stdin " ) / { ( ! % sep . ) + }
2018-06-19 01:27:32 -07:00
] ] , {
2018-07-24 15:25:03 -07:00
[ " true " ] = ( function ( )
2018-06-19 01:27:32 -07:00
return true
2018-07-24 15:25:03 -07:00
end ) ,
tonumber = ( function ( self )
return tonumber ( self )
end ) ,
2018-07-21 14:43:49 -07:00
sep = lpeg.P ( sep ) ,
2018-07-09 16:58:46 -07:00
add_file = function ( f )
return table.insert ( file_queue , f )
2018-07-21 14:43:49 -07:00
end ,
add_exec_string = function ( pos , s )
local name = " command line arg @ " .. tostring ( pos ) .. " .nom "
Files.spoof ( name , s )
return table.insert ( file_queue , name )
2017-09-12 21:38:54 -07:00
end
2018-06-19 01:27:32 -07:00
} )
2018-07-21 14:43:49 -07:00
local arg_string = table.concat ( arg , sep ) .. sep
2018-06-23 00:57:31 -07:00
local args = parser : match ( arg_string )
2018-06-19 01:27:32 -07:00
if not args or args.help then
print ( usage )
2018-06-21 19:12:59 -07:00
os.exit ( EXIT_FAILURE )
2018-06-19 01:27:32 -07:00
end
local nomsu = NomsuCompiler
2018-07-09 17:00:41 -07:00
nomsu.arg = NomsuCompiler.list ( args.nomsu_args )
2018-06-23 00:57:31 -07:00
if args.version then
2018-06-24 16:11:08 -07:00
nomsu : run ( [ [ use " core "
say ( Nomsu version ) ] ] )
2018-06-23 00:57:31 -07:00
os.exit ( EXIT_SUCCESS )
end
2018-06-21 19:12:59 -07:00
FILE_CACHE = setmetatable ( { } , {
__index = function ( self , filename )
local file = io.open ( filename )
if not ( file ) then
return nil
end
local contents = file : read ( " *a " )
file : close ( )
self [ filename ] = contents
return contents
end
} )
2018-06-19 01:27:32 -07:00
local run
run = function ( )
local input_files = { }
2018-07-09 16:58:46 -07:00
for _index_0 = 1 , # file_queue do
local f = file_queue [ _index_0 ]
2018-07-21 14:43:49 -07:00
if not ( Files.exists ( f ) ) then
error ( " Could not find: ' " .. tostring ( f ) .. " ' " )
2018-07-09 16:58:46 -07:00
end
2018-07-21 14:43:49 -07:00
for _ , filename in Files.walk ( f ) do
2018-07-09 16:58:46 -07:00
input_files [ filename ] = true
2018-06-19 01:27:32 -07:00
end
end
2018-06-23 00:57:31 -07:00
nomsu.can_optimize = function ( f )
2018-07-24 15:25:03 -07:00
if args.optimization == 0 then
2018-06-23 00:57:31 -07:00
return false
end
2018-07-09 16:58:46 -07:00
if args.compile and input_files [ f ] then
2018-06-23 00:57:31 -07:00
return false
end
return true
end
2018-07-23 14:53:12 -07:00
if not ( args.no_core ) then
for _ , filename in Files.walk ( ' core ' ) do
if filename : match ( " %.nom$ " ) then
nomsu : run_file ( filename )
end
end
end
2018-07-09 16:58:46 -07:00
local get_file_and_source
get_file_and_source = function ( filename )
local file , source
if filename == ' stdin ' then
file = io.read ( " *a " )
2018-07-21 14:43:49 -07:00
Files.spoof ( ' stdin ' , file )
2018-07-09 16:58:46 -07:00
source = Source ( ' stdin ' , 1 , # file )
elseif filename : match ( " %.nom$ " ) then
2018-07-21 14:43:49 -07:00
file = Files.read ( filename )
2018-07-09 16:58:46 -07:00
if not file then
error ( " File does not exist: " .. tostring ( filename ) , 0 )
end
source = Source ( filename , 1 , # file )
else
return nil
end
source = Source ( filename , 1 , # file )
return file , source
end
local run_file
run_file = function ( filename , lua_handler )
if lua_handler == nil then
lua_handler = nil
end
local file , source = get_file_and_source ( filename )
if not ( file ) then
2018-07-10 17:10:37 -07:00
return
2018-07-09 16:58:46 -07:00
end
local tree = nomsu : parse ( file , source )
if tree then
2018-07-15 19:41:22 -07:00
if tree.type ~= " FileChunks " then
2018-07-09 16:58:46 -07:00
tree = {
tree
}
end
for _index_0 = 1 , # tree do
local chunk = tree [ _index_0 ]
local lua = nomsu : compile ( chunk ) : as_statements ( " return " )
lua : declare_locals ( )
lua : prepend ( " -- File: " .. tostring ( source.filename : gsub ( " \n .* " , " ... " ) ) .. " \n " )
2018-07-10 17:10:37 -07:00
if lua_handler and input_files [ filename ] then
2018-07-09 16:58:46 -07:00
lua_handler ( tostring ( lua ) )
end
nomsu : run_lua ( lua )
end
end
end
2018-06-19 01:27:32 -07:00
local parse_errs = { }
2018-07-09 16:58:46 -07:00
for _index_0 = 1 , # file_queue do
local f = file_queue [ _index_0 ]
2018-07-21 14:43:49 -07:00
for _ , filename in Files.walk ( f ) do
2018-06-24 16:11:08 -07:00
local _continue_0 = false
repeat
2018-07-10 17:10:37 -07:00
if not ( filename == " stdin " or filename : match ( " %.nom$ " ) ) then
_continue_0 = true
break
end
2018-07-09 16:58:46 -07:00
if args.check_syntax then
local file , source = get_file_and_source ( filename )
if not ( file ) then
_continue_0 = true
break
2018-06-24 16:11:08 -07:00
end
2018-07-09 16:58:46 -07:00
nomsu : parse ( file , source )
print ( " Parse succeeded: " .. tostring ( filename ) )
2018-06-24 16:11:08 -07:00
end
if args.compile then
2018-07-09 16:58:46 -07:00
local output
if filename == ' stdin ' then
output = io.output ( )
2018-06-24 16:11:08 -07:00
else
2018-07-09 16:58:46 -07:00
output = io.open ( filename : gsub ( " %.nom$ " , " .lua " ) , " w " )
end
run_file ( filename , function ( lua )
output : write ( tostring ( lua ) , " \n " )
2018-06-24 16:11:08 -07:00
if args.verbose then
2018-07-09 16:58:46 -07:00
return print ( tostring ( lua ) )
2018-06-24 16:11:08 -07:00
end
2018-07-09 16:58:46 -07:00
end )
2018-06-24 16:11:08 -07:00
print ( ( " Compiled %-25s -> %s " ) : format ( filename , filename : gsub ( " %.nom$ " , " .lua " ) ) )
output : close ( )
end
2018-07-09 16:58:46 -07:00
if not args.check_syntax and not args.compile then
run_file ( filename , ( args.verbose and print or nil ) )
end
2018-06-24 16:11:08 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
end
end
2018-07-09 16:58:46 -07:00
if # file_queue == 0 then
2018-07-17 14:12:11 -07:00
nomsu : run ( [ [ # ! / usr / bin / env nomsu - V2
2018-06-23 18:26:18 -07:00
use " lib/consolecolor.nom "
2018-06-24 18:20:11 -07:00
action [ quit , exit ] : lua > " os.exit(0) "
2018-07-17 14:12:11 -07:00
action [ help ] :
2018-06-24 18:20:11 -07:00
say " .. "
This is the Nomsu v \ ( Nomsu version ) interactive console .
You can type in Nomsu code here and hit ' enter ' twice to run it .
To exit , type ' exit ' or ' quit ' and hit enter twice .
2018-06-23 18:26:18 -07:00
say " .. "
\ ( bright ) \ ( underscore ) Welcome to the Nomsu v \ ( Nomsu version ) interactive console ! \ ( reset color )
2018-06-24 18:20:11 -07:00
press ' enter ' twice to run a command
2018-06-23 18:26:18 -07:00
\ ( " " ) ] ] )
2018-06-19 01:27:32 -07:00
for repl_line = 1 , math.huge do
io.write ( colored.bright ( colored.yellow ( " >> " ) ) )
local buff = { }
while true do
2018-06-23 00:57:31 -07:00
io.write ( colors.bright )
2018-06-19 01:27:32 -07:00
local line = io.read ( " *L " )
2018-06-23 00:57:31 -07:00
io.write ( colors.reset )
2018-06-19 01:27:32 -07:00
if line == " \n " or not line then
if # buff > 0 then
io.write ( " \027 [1A \027 [2K " )
2018-04-28 18:07:14 -07:00
end
2018-06-19 01:27:32 -07:00
break
2018-04-28 18:07:14 -07:00
end
2018-06-19 01:27:32 -07:00
line = line : gsub ( " \t " , " " )
table.insert ( buff , line )
io.write ( colored.dim ( colored.yellow ( " .. " ) ) )
2018-06-15 03:11:38 -07:00
end
2018-06-19 01:27:32 -07:00
if # buff == 0 then
break
2018-04-28 18:07:14 -07:00
end
2018-06-19 01:27:32 -07:00
buff = table.concat ( buff )
2018-06-19 15:24:24 -07:00
local pseudo_filename = " user input # " .. repl_line
2018-07-21 14:43:49 -07:00
Files.spoof ( pseudo_filename , buff )
2018-06-19 01:27:32 -07:00
local err_hand
err_hand = function ( error_message )
2018-06-19 15:24:24 -07:00
return Errhand.print_error ( error_message )
2018-04-28 18:07:14 -07:00
end
2018-06-21 19:12:59 -07:00
local ret
ok , ret = xpcall ( nomsu.run , err_hand , nomsu , buff , Source ( pseudo_filename , 1 , # buff ) )
2018-06-19 01:27:32 -07:00
if ok and ret ~= nil then
print ( " = " .. repr ( ret ) )
elseif not ok then
2018-06-19 15:24:24 -07:00
Errhand.print_error ( ret )
2018-04-28 18:07:14 -07:00
end
end
end
2017-09-12 21:38:54 -07:00
end
2018-07-17 16:15:44 -07:00
local debugger
if args.debugger == " nil " then
debugger = { }
else
debugger = require ( args.debugger or ' error_handling ' )
end
local guard
if type ( debugger ) == ' function ' then
guard = debugger
else
guard = debugger.guard or debugger.call or debugger.wrap or debugger.run or ( function ( fn )
return fn ( )
end )
2018-06-19 02:00:52 -07:00
end
2018-07-17 16:15:44 -07:00
return guard ( run )