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
end ) ( ) , " ; " )
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-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
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
2018-06-23 17:22:23 -07:00
local files = require ( " files " )
2018-06-19 01:27:32 -07:00
local nomsu = NomsuCompiler
nomsu.arg = 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 ( )
for i , input in ipairs ( args.inputs ) do
if input == " - " then
2018-06-28 14:12:24 -07:00
args.inputs [ i ] = ' stdin '
2018-06-19 01:27:32 -07:00
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
2018-06-28 14:12:24 -07:00
local _continue_0 = false
repeat
local input = _list_0 [ _index_0 ]
if input == ' stdin ' then
input_files [ # input_files + 1 ] = ' stdin '
to_run [ ' stdin ' ] = true
_continue_0 = true
break
end
local found = false
for f in files.walk ( input ) do
input_files [ # input_files + 1 ] = f
to_run [ f ] = true
found = true
end
if not found then
error ( " Could not find: " .. tostring ( input ) )
end
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
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
local parse_errs = { }
2018-06-24 16:11:08 -07:00
local _list_1 = args.inputs
for _index_0 = 1 , # _list_1 do
local arg = _list_1 [ _index_0 ]
for filename in files.walk ( arg ) do
local _continue_0 = false
repeat
local file , source
2018-06-28 14:12:24 -07:00
if filename == ' stdin ' then
file = io.read ( " *a " )
2018-06-24 16:11:08 -07:00
files.spoof ( ' stdin ' , file )
source = Source ( ' stdin ' , 1 , # file )
elseif filename : match ( " %.nom$ " ) then
file = files.read ( filename )
if not file then
error ( " File does not exist: " .. tostring ( filename ) , 0 )
end
source = Source ( filename , 1 , # file )
else
_continue_0 = true
break
end
source = Source ( filename , 1 , # file )
local output
if args.compile then
output = io.open ( filename : gsub ( " %.nom$ " , " .lua " ) , " w " )
else
output = nil
end
if args.syntax then
local err
ok , err = pcall ( nomsu.parse , nomsu , file , source )
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
_continue_0 = true
break
end
local tree = nomsu : parse ( file , source )
if args.format then
local formatted = tree and tostring ( nomsu : tree_to_nomsu ( tree ) ) or " "
if print_file then
print_file : write ( formatted , " \n " )
print_file : flush ( )
end
_continue_0 = true
break
end
if tree then
if tree.type == " FileChunks " then
for _index_1 = 1 , # tree do
local chunk = tree [ _index_1 ]
local lua = nomsu : compile ( chunk ) : as_statements ( " return " )
lua : declare_locals ( )
lua : prepend ( " -- File: " .. tostring ( source.filename : gsub ( " \n .* " , " ... " ) ) .. " \n " )
if args.compile then
output : write ( tostring ( lua ) , " \n " )
end
if args.verbose then
print ( tostring ( lua ) )
end
nomsu : run_lua ( lua )
end
else
local lua = nomsu : compile ( tree ) : as_statements ( " return " )
lua : declare_locals ( )
lua : prepend ( " -- File: " .. tostring ( source.filename : gsub ( " \n .* " , " ... " ) ) .. " \n " )
if args.compile then
output : write ( tostring ( lua ) , " \n " )
end
if args.verbose then
print ( tostring ( lua ) )
end
nomsu : run_lua ( lua )
end
end
if args.compile then
print ( ( " Compiled %-25s -> %s " ) : format ( filename , filename : gsub ( " %.nom$ " , " .lua " ) ) )
output : close ( )
end
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
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 18:26:18 -07:00
nomsu : run ( [ [ use " core "
use " lib/consolecolor.nom "
2018-06-24 18:20:11 -07:00
action [ quit , exit ] : lua > " os.exit(0) "
action [ help ]
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-06-23 17:22:23 -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-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