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-06-23 00:57:31 -07:00
Usage : ( lua nomsu.lua | moon nomsu.moon ) [ - V version ] [ - i ] [ - O ] [ - v ] [ - c ] [ - f ] [ - s ] [ --help] [--version] [-p print_file] file1 file2... [-- nomsu args...]
2018-04-28 17:08:28 -07:00
OPTIONS
- i Run the compiler in interactive mode ( REPL )
- O Run the compiler in optimized mode ( use precompiled . lua versions of Nomsu files , when available )
2018-06-15 03:11:38 -07:00
- v Verbose : print compiled lua code
- c Compile . nom files into . lua files
2018-04-28 17:08:28 -07:00
- f Auto - format the given Nomsu file and print the result .
- s Check the program for syntax errors .
- h /-- help Print this message .
2018-06-23 00:57:31 -07:00
--version Print the version number and exit.
- V specify which Nomsu version is desired
2018-04-28 17:08:28 -07:00
- p < file > Print to the specified file instead of stdout .
2018-04-28 19:16:39 -07:00
< input > Input file 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-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
local STDIN , STDOUT , STDERR = " /dev/fd/0 " , " /dev/fd/1 " , " /dev/fd/2 "
if not arg or debug.getinfo ( 2 ) . func == require then
return NomsuCompiler
end
local parser = re.compile ( [ [ args <- { | ( flag " ; " ) * { : inputs : { | ( { file } " ; " ) * | } : } { : nomsu_args : { | ( " --; " ( { [ ^ ; ] * } " ; " ) * ) ? | } : } " ; " ? | } ! .
flag <-
{ : interactive : ( " -i " -> true ) : }
/ { : optimized : ( " -O " -> true ) : }
/ { : format : ( " -f " -> true ) : }
/ { : syntax : ( " -s " -> true ) : }
/ { : print_file : " -p " " ; " { file } : }
/ { : compile : ( " -c " -> true ) : }
/ { : verbose : ( " -v " -> true ) : }
/ { : help : ( ( " -h " / " --help " ) -> true ) : }
2018-06-23 00:57:31 -07:00
/ { : version : ( " --version " -> true ) : }
/ { : requested_version : " -V " ( ( " ; " ) ? { ( [ 0 - 9. ] ) + } ) ? : }
2018-06-19 01:27:32 -07:00
file <- " - " / [ ^ ; ] +
] ] , {
[ " true " ] = function ( )
return true
2017-09-12 21:38:54 -07:00
end
2018-06-19 01:27:32 -07:00
} )
2018-06-23 00:57:31 -07:00
local arg_string = table.concat ( arg , " ; " ) .. " ; "
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
nomsu.arg = args.nomsu_args
2018-06-23 00:57:31 -07:00
if args.version then
nomsu : run ( ' use "core" \n say (Nomsu version) ' )
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
} )
local match , sub , rep , gsub , format , byte , find
do
local _obj_0 = string
match , sub , rep , gsub , format , byte , match , find = _obj_0.match , _obj_0.sub , _obj_0.rep , _obj_0.gsub , _obj_0.format , _obj_0.byte , _obj_0.match , _obj_0.find
end
local iterate_single
iterate_single = function ( item , prev )
if item == prev then
return nil
else
return item
end
end
local lfs
ok , lfs = pcall ( require , " lfs " )
if ok then
all_files = function ( path )
local browse
browse = function ( filename )
local file_type = lfs.attributes ( filename , ' mode ' )
if file_type == ' file ' then
if match ( filename , " %.nom$ " ) or match ( filename , " %.lua$ " ) then
coroutine.yield ( filename )
return true
end
elseif file_type == ' directory ' then
for subfile in lfs.dir ( filename ) do
if not ( subfile == " . " or subfile == " .. " ) then
browse ( filename .. " / " .. subfile )
end
end
return true
elseif file_type == ' char device ' then
coroutine.yield ( filename )
return true
end
return false
end
return coroutine.wrap ( function ( )
if not browse ( path ) and package.nomsupath then
browse ( package.nomsupath .. " / " .. path )
end
return nil
end )
end
else
local ret = os.execute ( ' find . -maxdepth 0 ' )
if not ( ret == true or ret == 0 ) then
error ( " Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: http://keplerproject.github.io/luafilesystem/ or `luarocks install luafilesystem`) " , 0 )
end
all_files = function ( path )
if match ( path , " %.nom$ " ) or match ( path , " %.lua$ " ) or match ( path , " ^/dev/fd/[012]$ " ) then
return iterate_single , path
end
path = gsub ( path , " \\ " , " \\ \\ " )
path = gsub ( path , " ` " , " " )
path = gsub ( path , ' " ' , ' \\ " ' )
path = gsub ( path , " $ " , " " )
return coroutine.wrap ( function ( )
local f = io.popen ( ' find -L " ' .. path .. ' " -not -path "*/ \\ .*" -type f -name "*.nom" ' )
local found = false
for line in f : lines ( ) do
found = true
coroutine.yield ( line )
end
if not found and package.nomsupath then
f : close ( )
f = io.popen ( ' find -L " ' .. package.nomsupath .. ' / ' .. path .. ' " -not -path "*/ \\ .*" -type f -name "*.nom" ' )
for line in f : lines ( ) do
coroutine.yield ( line )
end
end
local success = f : close ( )
if not ( success ) then
return error ( " Invalid file path: " .. tostring ( path ) )
end
end )
end
end
2018-06-19 01:27:32 -07:00
local run
run = function ( )
for i , input in ipairs ( args.inputs ) do
if input == " - " then
args.inputs [ i ] = STDIN
end
end
if # args.inputs == 0 and not args.interactive then
args.inputs = {
" core "
}
args.interactive = true
end
local print_file
if args.print_file == " - " then
print_file = io.stdout
elseif args.print_file then
print_file = io.open ( args.print_file , ' w ' )
else
print_file = io.stdout
end
if print_file == nil then
nomsu.print = function ( ) end
elseif print_file ~= io.stdout then
nomsu.print = function ( ... )
local N = select ( " # " , ... )
if N > 0 then
print_file : write ( tostring ( select ( 1 , ... ) ) )
for i = 2 , N do
print_file : write ( ' \t ' , tostring ( select ( 1 , ... ) ) )
end
end
print_file : write ( ' \n ' )
return print_file : flush ( )
end
end
local input_files = { }
local to_run = { }
local _list_0 = args.inputs
for _index_0 = 1 , # _list_0 do
local input = _list_0 [ _index_0 ]
for f in all_files ( input ) do
input_files [ # input_files + 1 ] = f
to_run [ f ] = true
end
end
2018-06-23 00:57:31 -07:00
nomsu.can_optimize = function ( f )
if not ( args.optimized ) then
return false
end
if to_run [ f ] then
return false
end
return true
end
2018-06-19 01:27:32 -07:00
if args.compile or args.verbose then
nomsu.on_compile = function ( code , from_file )
if not ( to_run [ from_file ] ) then
return
end
if args.verbose then
io.write ( tostring ( code ) , " \n " )
end
if args.compile and from_file : match ( " %.nom$ " ) then
local output_filename = from_file : gsub ( " %.nom$ " , " .lua " )
local output_file = io.open ( output_filename , ' w ' )
output_file : write ( tostring ( code ) )
output_file : flush ( )
print ( ( " Compiled %-25s -> %s " ) : format ( from_file , output_filename ) )
return output_file : close ( )
end
end
end
local parse_errs = { }
for _index_0 = 1 , # input_files do
local filename = input_files [ _index_0 ]
if args.syntax then
local file_contents = io.open ( filename ) : read ( ' *a ' )
2018-06-21 19:12:59 -07:00
local err
ok , err = pcall ( nomsu.parse , nomsu , file_contents , Source ( filename , 1 , # file_contents ) )
2018-06-19 01:27:32 -07:00
if not ok then
table.insert ( parse_errs , err )
elseif print_file then
print_file : write ( " Parse succeeded: " .. tostring ( filename ) .. " \n " )
print_file : flush ( )
end
elseif args.format then
local file = FILE_CACHE [ filename ]
if not file then
error ( " File does not exist: " .. tostring ( filename ) , 0 )
end
local tree = nomsu : parse ( file , Source ( filename , 1 , # file ) )
local formatted = tostring ( nomsu : tree_to_nomsu ( tree ) )
if print_file then
print_file : write ( formatted , " \n " )
print_file : flush ( )
end
elseif filename == STDIN then
local file = io.input ( ) : read ( " *a " )
FILE_CACHE.stdin = file
nomsu : run ( file , Source ( ' stdin ' , 1 , # file ) )
2018-04-28 18:07:14 -07:00
else
2018-06-19 01:27:32 -07:00
nomsu : run_file ( filename )
end
end
if # parse_errs > 0 then
io.stderr : write ( table.concat ( parse_errs , " \n \n " ) )
io.stderr : flush ( )
2018-06-21 19:12:59 -07:00
os.exit ( EXIT_FAILURE )
2018-06-19 01:27:32 -07:00
elseif args.syntax then
2018-06-21 19:12:59 -07:00
os.exit ( EXIT_SUCCESS )
2018-06-19 01:27:32 -07:00
end
if args.interactive then
2018-06-23 00:57:31 -07:00
nomsu : run ( ' say " \\ n \\ (bright) \\ (underscore)Welcome to the Nomsu v \\ (Nomsu version) interactive console! \\ (reset color) \\ n press \' enter \' twice to run a command \\ n" ' )
local ready_to_quit = false
nomsu.A_quit = function ( )
ready_to_quit = true
return print ( " Goodbye! " )
end
nomsu.A_exit = nomsu.A_quit
nomsu.A_help = function ( )
print ( " This is the Nomsu v " .. tostring ( nomsu.A_Nomsu_version ( ) ) .. " interactive console. " )
print ( " You can type in Nomsu code here and hit 'enter' twice to run it. " )
return print ( " To exit, type 'exit' or 'quit' and hit enter twice " )
end
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
FILE_CACHE [ 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
2018-06-23 00:57:31 -07:00
if ready_to_quit then
break
end
2018-04-28 18:07:14 -07:00
end
end
2017-09-12 21:38:54 -07:00
end
2018-06-19 02:00:52 -07:00
local has_ldt , ldt = pcall ( require , ' ldt ' )
if has_ldt then
return ldt.guard ( run )
else
2018-06-19 15:24:24 -07:00
return Errhand.run_safely ( run )
2018-06-19 02:00:52 -07:00
end