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 if NOMSU_VERSION and NOMSU_PREFIX then 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] _accum_0[_len_0] = tostring(NOMSU_PREFIX) .. "/share/nomsu/" .. tostring(v) .. "/?.lua" _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] _accum_0[_len_0] = tostring(NOMSU_PREFIX) .. "/lib/nomsu/" .. tostring(v) .. "/?.so" _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] _accum_0[_len_0] = tostring(NOMSU_PREFIX) .. "/share/nomsu/" .. tostring(v) _len_0 = _len_0 + 1 end return _accum_0 end)(), ";") .. ";." else package.nomsupath = "." end local usage = [=[Nomsu Compiler Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [-O optimization level] [-v] [-c] [-s] [-I file] [--help | -h] [--version] [--no-core] [file [nomsu args...]] OPTIONS -O Run the compiler with the given optimization level (>0: use precompiled .lua versions of Nomsu files, when available). -v Verbose: print compiled lua code. -c Compile the input files into a .lua files. -e Execute the specified string. -s Check the input files for syntax errors. -I Add an additional input file or directory. -d Attempt to use the specified debugger to wrap the main body of execution. -h/--help Print this message. --version Print the version number and exit. --no-core Skip loading the Nomsu core by default. -V specify which Nomsu version is desired. The Nomsu file to run (can be "-" to use stdin). ]=] 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 local Files = require("files") local Errhand = require("error_handling") local NomsuCode, LuaCode, Source do local _obj_0 = require("code_obj") NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source end 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') if not arg or debug.getinfo(2).func == require then return nomsu_environment end local file_queue = List({ }) local sep = "\3" local parser = re.compile([[ args <- {| (flag %sep)* (({~ file ~} -> add_file) {:primary_file: %true :} %sep)? {:nomsu_args: {| (nomsu_flag %sep)* {:extra_args: {| ({[^%sep]+} %sep)* |} :} |} :} |} !. flag <- {:optimization: "-O" (%sep? %number)? :} / ("-I" %sep? ({~ file ~} -> add_file)) / ("-e" %sep? (({} {~ file ~}) -> add_exec_string) {:exec_strings: %true :}) / ({:check_syntax: "-s" %true:}) / ({:compile: "-c" %true:}) / {:verbose: "-v" %true :} / {:help: ("-h" / "--help") %true :} / {:version: "--version" %true :} / {:no_core: "--no-core" %true :} / {:debugger: ("-d" %sep? {(!%sep .)*}) :} / {:requested_version: "-V" (%sep? {([0-9.])+})? :} nomsu_flag <- {| ({:key: ('-' [a-z]) :} {:value: %true :}) / ({:key: ('--' [^%sep=]+) :} {:value: ('=' {[^%sep]+}) / %true :}) |} file <- ("-" -> "stdin") / {(!%sep .)+} ]], { ["true"] = lpeg.Cc(true), number = lpeg.R("09") ^ 1 / tonumber, sep = lpeg.P(sep), add_file = function(f) return file_queue:add(f) end, add_exec_string = function(pos, s) local name = "command line arg @" .. tostring(pos) .. ".nom" Files.spoof(name, s) return file_queue:add(name) end }) local arg_string = table.concat(arg, sep) .. sep local args = parser:match(arg_string) if not args or args.help then print(usage) os.exit(EXIT_FAILURE) end 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 nomsu_environment.OPTIMIZATION = tonumber(args.optimization or 1) if args.version then nomsu_environment.run_file_1_in('core', nomsu_environment, nomsu_environment.OPTIMIZATION) nomsu_environment.run_1_in([[say (Nomsu version)]], nomsu_environment) os.exit(EXIT_SUCCESS) end local run run = function() local input_files = { } for _index_0 = 1, #file_queue do local f = file_queue[_index_0] if not (Files.exists(f)) then error("Could not find: '" .. tostring(f) .. "'") end local _list_1 = Files.list(f) for _index_1 = 1, #_list_1 do local filename = _list_1[_index_1] input_files[filename] = true end end if not (args.no_core) then nomsu_environment.run_file_1_in('core', nomsu_environment, nomsu_environment.OPTIMIZATION) end for _index_0 = 1, #file_queue do local _continue_0 = false repeat local filename = file_queue[_index_0] if not (filename == "stdin" or filename:match("%.nom$")) then _continue_0 = true break end 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 print(lua:text()) end 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 } 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") print(lua:text()) nomsu_environment.run_1_in(lua, nomsu_environment) end else nomsu_environment.run_file_1_in(filename, nomsu_environment, 0) end _continue_0 = true until true if not _continue_0 then break end end if not (args.primary_file or args.exec_strings) then return nomsu_environment.run_file_1_in("tools/repl.nom", nomsu_environment, nomsu_environment.OPTIMIZATION) end end 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) end return guard(run)