2019-02-06 15:33:10 -08:00
NOMSU_VERSION = {
7 ,
0 ,
2019-09-20 14:40:15 -07:00
1
2019-02-06 15:33:10 -08:00
}
2019-01-27 02:07:12 -08:00
local clibtype = package.cpath : match ( " ?%.(so) " ) or package.cpath : match ( " ?%.(dll) " )
2019-02-06 15:33:10 -08:00
if NOMSU_PREFIX then
package.path = tostring ( NOMSU_PREFIX ) .. " /share/nomsu/ " .. tostring ( table.concat ( NOMSU_VERSION , " . " ) ) .. " /?.lua; " .. package.path
package.cpath = tostring ( NOMSU_PREFIX ) .. " /lib/nomsu/ " .. tostring ( table.concat ( NOMSU_VERSION , " . " ) ) .. " /?. " .. tostring ( clibtype ) .. " ; " .. package.cpath
end
2019-01-29 16:16:45 -08:00
COLOR_ENABLED = true
2019-01-27 02:07:12 -08:00
if clibtype == " dll " then
2019-05-02 15:42:00 -07:00
local ok , enable_colors = pcall ( require , ' wincolors ' )
COLOR_ENABLED = false
if ok then
local _
ok , _ = pcall ( enable_colors )
if ok then
COLOR_ENABLED = false
end
2019-01-29 16:16:45 -08:00
end
2019-05-02 15:42:00 -07:00
pcall ( os.execute , " chcp 65001>nul " )
2019-01-27 02:07:12 -08:00
end
2019-02-06 15:33:10 -08:00
local List , Dict
do
local _obj_0 = require ( ' containers ' )
List , Dict = _obj_0.List , _obj_0.Dict
2019-01-10 16:33:37 -08:00
end
2019-02-06 15:33:10 -08:00
NOMSU_VERSION = List ( NOMSU_VERSION )
local Text = require ( ' text ' )
require ( ' builtin_metatables ' )
2019-09-20 14:40:15 -07:00
local EXIT_FAILURE = 1
local EXIT_SUCCESS = 0
local usage = [ = [ Nomsu Usage : nomsu [ - V version ] [ --help | -h] [--version] [-O optimization level] [-v] [-c] [-s] [-d debugger] [--no-core] [(file | -t tool | -e "nomsu code..." | 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 .
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 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-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 ) *
2019-01-10 16:33:37 -08:00
{ : files : { |
2019-01-14 15:42:48 -08:00
( ( " -m " % sep ) ? ( file % sep ) + " -- " % sep
/ file % sep
2019-01-10 16:33:37 -08:00
/ { ~ ' ' % sep ? -> ' nomsu://tools/repl.nom ' ~ } ) | } : }
2018-12-18 19:23:20 -08:00
{ : nomsu_args : { | ( nomsu_flag % sep ) * { : extras : { | ( { [ ^% sep ] + } % sep ) * | } : } | } : }
2019-01-14 15:42:48 -08:00
| } ( { . + } ? )
2018-12-31 01:37:24 -08:00
2019-01-10 16:33:37 -08:00
file <-
2019-01-14 15:42:48 -08:00
( " -e " % sep ( { [ ^% sep ] + } -> spoof )
/ " -t " % sep { ~ { [ ^% sep ] + } -> " nomsu://tools/%1.nom " ~ }
/ ! " -- " { [ ^% sep ] + } )
2019-01-10 16:33:37 -08:00
2018-12-31 01:37:24 -08:00
flag <- longflag / shortflag / " - " shortboolflag +
longflag <-
{ : help : " --help " % true : }
2018-09-12 15:31:59 -07:00
/ { : version : " --version " % true : }
/ { : no_core : " --no-core " % true : }
2018-12-31 01:37:24 -08:00
shortflag <-
{ : optimization : " -O " % sep ? % number : }
/ { : debugger : ( " -d " % sep ? { [ ^% sep ] + } ) : }
/ { : requested_version : " -V " % sep ? { ( [ 0 - 9. ] ) + } : }
shortboolflag <-
{ : check_syntax : " s " % true : }
/ { : compile : " c " % true : }
/ { : verbose : " v " % true : }
/ { : help : " h " % true : }
nomsu_flag <- nomsu_longflag / " - " nomsu_shortboolflag +
nomsu_shortboolflag <- { | { : key : [ a - zA - Z ] : } { : value : % true : } | }
nomsu_longflag <- ' -- ' { | { : 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 ,
2019-01-10 16:33:37 -08:00
sep = lpeg.P ( sep ) ,
spoof = Files.spoof
2018-06-19 01:27:32 -07:00
} )
2018-07-21 14:43:49 -07:00
local arg_string = table.concat ( arg , sep ) .. sep
2019-01-14 15:42:48 -08:00
local args , err = parser : match ( arg_string )
2019-09-20 14:40:15 -07:00
if not args or err then
2019-01-14 15:42:48 -08:00
if err then
2019-01-29 16:16:45 -08:00
print ( " Didn't understand: " .. tostring ( err ) )
2019-01-14 15:42:48 -08:00
end
2018-06-19 01:27:32 -07:00
print ( usage )
2018-06-21 19:12:59 -07:00
os.exit ( EXIT_FAILURE )
2018-06-19 01:27:32 -07:00
end
2019-09-20 14:40:15 -07:00
if args.help then
print ( " Nomsu - A dynamically typed programming language with natural syntax and strong metaprogramming abilities. " )
print ( " https://nomsu.org \n " )
print ( usage )
os.exit ( EXIT_SUCCESS )
end
2019-02-06 15:33:10 -08:00
if args.version then
print ( NOMSU_VERSION : joined_with ( " . " ) )
os.exit ( EXIT_SUCCESS )
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
2018-12-18 19:23:20 -08:00
nomsu_args.extras = List ( args.nomsu_args . extras or { } )
2019-01-10 16:33:37 -08:00
local optimization = tonumber ( args.optimization or 1 )
local nomsupath = { }
2019-01-14 15:42:48 -08:00
local suffixes
if optimization > 0 then
suffixes = {
" ?.lua " ,
" ?/init.lua " ,
" ?.nom " ,
" ?/init.nom "
}
else
suffixes = {
" ?.nom " ,
" ?/init.nom "
}
2019-01-10 16:33:37 -08:00
end
2019-01-14 15:42:48 -08:00
local add_path
add_path = function ( p )
for _index_0 = 1 , # suffixes do
local s = suffixes [ _index_0 ]
table.insert ( nomsupath , p .. " / " .. s )
2019-01-10 16:33:37 -08:00
end
end
2019-02-06 15:33:10 -08:00
if NOMSU_PREFIX then
add_path ( tostring ( NOMSU_PREFIX ) .. " /share/nomsu/ " .. tostring ( NOMSU_VERSION : joined_with ( " . " ) ) .. " /lib " )
2019-01-14 15:42:48 -08:00
else
add_path ( " ./lib " )
2019-01-10 16:33:37 -08:00
end
2019-01-14 17:59:39 -08:00
local NOMSU_PACKAGEPATH = NOMSU_PACKAGEPATH or " /opt/nomsu "
2019-01-14 15:42:48 -08:00
add_path ( NOMSU_PACKAGEPATH )
add_path ( " . " )
2019-01-10 16:33:37 -08:00
package.nomsupath = table.concat ( nomsupath , " ; " )
2019-01-15 15:53:31 -08:00
package.nomsuloaded = Dict ( { } )
2019-01-10 16:33:37 -08:00
local nomsu_environment = require ( ' nomsu_environment ' )
2019-02-06 15:33:10 -08:00
nomsu_environment.NOMSU_VERSION = NOMSU_VERSION
2018-12-18 19:23:20 -08:00
nomsu_environment.COMMAND_LINE_ARGS = nomsu_args
2019-01-10 16:33:37 -08:00
nomsu_environment.OPTIMIZATION = optimization
nomsu_environment.NOMSU_PACKAGEPATH = NOMSU_PACKAGEPATH
nomsu_environment.NOMSU_PREFIX = NOMSU_PREFIX
2019-01-29 16:16:45 -08:00
nomsu_environment.COLOR_ENABLED = COLOR_ENABLED
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
2019-01-10 16:33:37 -08:00
nomsu_environment : export ( " core " )
2018-11-26 16:16:08 -08:00
end
local input_files = { }
2019-09-20 14:40:15 -07:00
if args.files and # args.files > 0 then
2018-11-26 16:16:08 -08:00
local _list_1 = args.files
for _index_0 = 1 , # _list_1 do
local f = _list_1 [ _index_0 ]
do
2019-01-10 16:33:37 -08:00
local nomsu_name = f : match ( " ^nomsu://(.*)%.nom " )
2018-11-26 16:16:08 -08:00
if nomsu_name then
2019-01-14 15:42:48 -08:00
local path
path , err = package.searchpath ( nomsu_name , package.nomsupath , " / " )
2019-01-10 16:33:37 -08:00
if not path then
error ( err )
2018-11-08 15:23:22 -08:00
end
2019-01-10 16:33:37 -08:00
table.insert ( input_files , path )
2018-11-26 16:16:08 -08:00
else
2019-01-10 16:33:37 -08:00
table.insert ( input_files , f )
2018-11-19 17:21:08 -08:00
end
2018-11-26 16:16:08 -08:00
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
2019-01-15 15:53:31 -08:00
local code = Files.read ( filename )
if not code then
error ( " Could not find file ' " .. tostring ( filename ) .. " ' " )
end
2019-01-14 18:00:59 -08:00
if filename : match ( " %.lua$ " ) then
error ( " Cannot compile a lua file (expected a nomsu file as input) " )
end
2018-11-26 16:16:08 -08:00
local output
if filename == ' stdin ' then
output = io.output ( )
else
output = io.open ( filename : gsub ( " %.nom$ " , " .lua " ) , " w " )
end
local source = Source ( filename , 1 , # code )
code = NomsuCode : from ( source , code )
2019-01-14 15:42:48 -08:00
local env = nomsu_environment.new_environment ( )
env.MODULE_NAME = filename
local tree = env._1_parsed ( code )
2019-02-05 15:45:27 -08:00
if tree.shebang then
output : write ( tree.shebang )
end
2018-11-26 16:16:08 -08:00
if not ( tree.type == ' FileChunks ' ) then
tree = {
tree
}
end
for chunk_no , chunk in ipairs ( tree ) do
2019-01-14 15:42:48 -08:00
local lua = env : compile ( chunk )
2018-11-26 16:16:08 -08:00
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
2019-01-14 15:42:48 -08:00
env : run ( chunk )
2018-11-26 16:16:08 -08:00
output : write ( lua : text ( ) , " \n " )
end
print ( ( " Compiled %-25s -> %s " ) : format ( filename , filename : gsub ( " %.nom$ " , " .lua " ) ) )
output : close ( )
elseif args.verbose then
2019-01-14 15:42:48 -08:00
local env = nomsu_environment.new_environment ( )
env.MODULE_NAME = filename
env.WAS_RUN_DIRECTLY = true
2019-01-14 17:59:39 -08:00
local code = Files.read ( filename )
local source = Source ( filename , 1 , # code )
if filename : match ( " %.lua$ " ) then
code = LuaCode : from ( Source ( filename , 1 , # code ) , code )
print ( code : text ( ) )
env : run ( code )
else
code = NomsuCode : from ( source , code )
local tree = env._1_parsed ( code )
if not ( tree.type == ' FileChunks ' ) then
tree = {
tree
}
end
for chunk_no , chunk in ipairs ( tree ) do
local lua = env : compile ( chunk )
lua : declare_locals ( )
lua : prepend ( ( chunk_no > 1 ) and ' \n ' or ' ' , " -- File " .. tostring ( filename ) .. " chunk # " .. tostring ( chunk_no ) .. " \n " )
print ( lua : text ( ) )
env : run ( lua )
end
2018-11-26 16:16:08 -08:00
end
else
2019-01-10 16:33:37 -08:00
local f = Files.read ( filename )
2019-09-20 14:40:15 -07:00
if not f then
if filename : match ( " ^- " ) then
print ( " Not a valid flag: " .. filename .. " \n " )
print ( usage )
else
print ( " File not found: " .. filename )
end
os.exit ( EXIT_FAILURE )
end
2019-01-10 16:33:37 -08:00
if filename : match ( " %.lua$ " ) then
f = LuaCode : from ( Source ( filename , 1 , # f ) , f )
end
2019-01-14 15:42:48 -08:00
local env = nomsu_environment.new_environment ( )
env.MODULE_NAME = filename
env.WAS_RUN_DIRECTLY = true
env : run ( f )
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 )