2018-07-24 16:43:06 -07:00
local EXIT_SUCCESS , EXIT_FAILURE = 0 , 1
if _VERSION == " Lua 5.1 " and not jit then
print ( " Sorry, Nomsu does not run on Lua 5.1. Please use LuaJIT 2+ or Lua 5.2+ " )
os.exit ( EXIT_FAILURE )
end
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-19 01:27:32 -07:00
local usage = [ = [ Nomsu Compiler
2018-04-28 17:08:28 -07:00
2018-11-26 16:16:08 -08:00
Usage : ( nomsu | lua nomsu.lua | moon nomsu.moon ) [ - V version ] [ --help | -h] [--version] [-O optimization level] [-v] [-c] [-s] [-d debugger] [--no-core] [(file | -t tool | -e "nomsu code..." | -m files... [--]) [nomsu args...]]
2018-04-28 17:08:28 -07:00
OPTIONS
2018-11-26 16:16:08 -08:00
- t < tool > Run a tool .
- e Execute the specified string .
- m Run multiple files ( all extra arguments ) .
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 .
- s Check the input files for syntax errors .
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 NomsuCode , LuaCode , Source
do
local _obj_0 = require ( " code_obj " )
NomsuCode , LuaCode , Source = _obj_0.NomsuCode , _obj_0.LuaCode , _obj_0.Source
end
2018-11-08 15:23:22 -08:00
local List , Dict , Text
do
local _obj_0 = require ( ' containers ' )
List , Dict , Text = _obj_0.List , _obj_0.Dict , _obj_0.Text
end
local nomsu_environment = require ( ' nomsu_environment ' )
2018-06-19 01:27:32 -07:00
if not arg or debug.getinfo ( 2 ) . func == require then
2018-11-08 15:23:22 -08:00
return nomsu_environment
2018-06-19 01:27:32 -07:00
end
2018-09-08 01:05:59 -07:00
local sep = " \3 "
2018-11-26 16:16:08 -08:00
local parser = re.compile ( [ [ args <- { | ( flag % sep ) *
( ( " -e " % sep { : execute : { [ ^% sep ] + } : } % sep )
/ { : files : { |
( " -t " % sep { ~ { [ ^% sep ] + } -> " nomsu://tools/%1.nom " ~ } % sep
/ " -m " % sep ( ! ( " -- " % sep ) { [ ^% sep ] + } % sep ) * ( " -- " % sep ) ?
/ { [ ^% sep ] + } % sep
/ { ~ ' ' % sep ? -> ' nomsu://tools/repl.nom ' ~ } ) | } : } )
{ : nomsu_args : { | ( nomsu_flag % sep ) * { : extra_args : { | ( { [ ^% sep ] + } % sep ) * | } : } | } : }
| } ! .
2018-06-19 01:27:32 -07:00
flag <-
2018-09-12 15:31:59 -07:00
{ : optimization : " -O " ( % sep ? % number ) ? : }
/ ( { : check_syntax : " -s " % true : } )
/ ( { : compile : " -c " % true : } )
/ { : verbose : " -v " % true : }
/ { : help : ( " -h " / " --help " ) % true : }
/ { : version : " --version " % true : }
/ { : no_core : " --no-core " % true : }
2018-11-26 16:16:08 -08:00
/ { : debugger : ( " -d " % sep ? { [ ^% sep ] * } ) : }
2018-07-21 14:43:49 -07:00
/ { : requested_version : " -V " ( % sep ? { ( [ 0 - 9. ] ) + } ) ? : }
2018-11-20 14:52:59 -08:00
nomsu_flag <- { | ( { : key : ( ' - ' [ a - z ] ) : } { : value : % true : } ) / ( { : key : ( ' -- ' [ ^% sep = ] + ) : } { : value : ( ' = ' { [ ^% sep ] + } ) / % true : } ) | }
2018-06-19 01:27:32 -07:00
] ] , {
2018-09-12 15:31:59 -07:00
[ " true " ] = lpeg.Cc ( true ) ,
number = lpeg.R ( " 09 " ) ^ 1 / tonumber ,
2018-11-26 16:16:08 -08:00
sep = lpeg.P ( sep )
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
2018-11-20 14:52:59 -08:00
local nomsu_args = Dict ( { } )
local _list_0 = args.nomsu_args
for _index_0 = 1 , # _list_0 do
local argpair = _list_0 [ _index_0 ]
nomsu_args [ argpair.key ] = argpair.value
end
nomsu_args.extra_args = List ( args.nomsu_args . extra_args or { } )
nomsu_environment.command_line_args = nomsu_args
2018-11-11 18:03:18 -08:00
nomsu_environment.OPTIMIZATION = tonumber ( args.optimization or 1 )
2018-06-19 01:27:32 -07:00
local run
run = function ( )
2018-07-23 14:53:12 -07:00
if not ( args.no_core ) then
2018-11-26 16:16:08 -08:00
for nomsupath in package.nomsupath : gmatch ( " [^;]+ " ) do
local _continue_0 = false
repeat
local files = Files.list ( nomsupath .. " /core " )
if not ( files ) then
_continue_0 = true
break
end
for _index_0 = 1 , # files do
local _continue_1 = false
repeat
local f = files [ _index_0 ]
if not ( f : match ( " %.nom$ " ) ) then
_continue_1 = true
break
end
nomsu_environment.run_file_1_in ( f , nomsu_environment , nomsu_environment.OPTIMIZATION )
_continue_1 = true
until true
if not _continue_1 then
break
end
end
2018-11-19 17:21:08 -08:00
_continue_0 = true
2018-11-26 16:16:08 -08:00
until true
if not _continue_0 then
2018-11-19 17:21:08 -08:00
break
end
2018-11-26 16:16:08 -08:00
end
end
if args.version then
nomsu_environment.run_1_in ( " say (Nomsu version) " , nomsu_environment )
os.exit ( EXIT_SUCCESS )
end
local input_files = { }
if args.execute then
table.insert ( input_files , Files.spoof ( " <input command> " , args.execute ) )
end
if args.files then
local _list_1 = args.files
for _index_0 = 1 , # _list_1 do
local f = _list_1 [ _index_0 ]
local files
do
local nomsu_name = f : match ( " ^nomsu://(.*) " )
if nomsu_name then
for nomsupath in package.nomsupath : gmatch ( " [^;]+ " ) do
local _continue_0 = false
repeat
files = Files.list ( nomsupath .. " / " .. nomsu_name )
if not ( files ) then
_continue_0 = true
break
end
_continue_0 = true
until true
if not _continue_0 then
break
end
2018-11-08 15:23:22 -08:00
end
2018-11-26 16:16:08 -08:00
else
files = Files.list ( f )
2018-11-19 17:21:08 -08:00
end
2018-11-26 16:16:08 -08:00
end
if not ( files and # files > 0 ) then
error ( " Could not find: ' " .. tostring ( f ) .. " ' " )
end
for _index_1 = 1 , # files do
local filename = files [ _index_1 ]
table.insert ( input_files , filename )
end
end
end
for _index_0 = 1 , # input_files do
local filename = input_files [ _index_0 ]
if args.check_syntax then
local code = Files.read ( filename )
local source = Source ( filename , 1 , # code )
nomsu_environment._1_parsed ( NomsuCode : from ( source , code ) )
print ( " Parse succeeded: " .. tostring ( filename ) )
elseif args.compile then
local output
if filename == ' stdin ' then
output = io.output ( )
else
output = io.open ( filename : gsub ( " %.nom$ " , " .lua " ) , " w " )
end
local code = Files.read ( filename )
local source = Source ( filename , 1 , # code )
code = NomsuCode : from ( source , code )
local tree = nomsu_environment._1_parsed ( code )
if not ( tree.type == ' FileChunks ' ) then
tree = {
tree
}
end
for chunk_no , chunk in ipairs ( tree ) do
local lua = nomsu_environment.compile ( chunk )
lua : declare_locals ( )
lua : prepend ( ( chunk_no > 1 ) and ' \n ' or ' ' , " -- File " .. tostring ( filename ) .. " chunk # " .. tostring ( chunk_no ) .. " \n " )
if args.verbose then
2018-11-19 17:21:08 -08:00
print ( lua : text ( ) )
end
2018-11-26 16:16:08 -08:00
nomsu_environment.run_1_in ( chunk , nomsu_environment )
output : write ( lua : text ( ) , " \n " )
end
print ( ( " Compiled %-25s -> %s " ) : format ( filename , filename : gsub ( " %.nom$ " , " .lua " ) ) )
output : close ( )
elseif args.verbose then
local code = Files.read ( filename )
local source = Source ( filename , 1 , # code )
code = NomsuCode : from ( source , code )
local tree = nomsu_environment._1_parsed ( code )
if not ( tree.type == ' FileChunks ' ) then
tree = {
tree
}
2018-06-19 01:27:32 -07:00
end
2018-11-26 16:16:08 -08:00
for chunk_no , chunk in ipairs ( tree ) do
local lua = nomsu_environment.compile ( chunk )
lua : declare_locals ( )
lua : prepend ( ( chunk_no > 1 ) and ' \n ' or ' ' , " -- File " .. tostring ( filename ) .. " chunk # " .. tostring ( chunk_no ) .. " \n " )
print ( lua : text ( ) )
nomsu_environment.run_1_in ( lua , nomsu_environment )
end
else
nomsu_environment.run_file_1_in ( filename , nomsu_environment , 0 )
2018-06-19 01:27:32 -07:00
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 )