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] [-t] [-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 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 local repr repr = require("utils").repr if not arg or debug.getinfo(2).func == require then return NomsuCompiler end local file_queue = { } local sep = "\0" local parser = re.compile([[ args <- {| (flag %sep)* (({~ file ~} -> add_file) %sep)? {:nomsu_args: {| ({(!%sep .)*} %sep)* |} :} %sep? |} !. flag <- {:optimization: "-O" (%sep? (([0-9]+) -> tonumber))? :} / ("-I" %sep? ({~ file ~} -> add_file)) / ("-e" %sep? (({} {~ file ~}) -> add_exec_string)) / ({:check_syntax: ("-s" -> true):}) / ({:compile: ("-c" -> 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.])+})? :} file <- ("-" -> "stdin") / {(!%sep .)+} ]], { ["true"] = (function() return true end), tonumber = (function(self) return tonumber(self) end), sep = lpeg.P(sep), add_file = function(f) return table.insert(file_queue, f) end, add_exec_string = function(pos, s) local name = "command line arg @" .. tostring(pos) .. ".nom" Files.spoof(name, s) return table.insert(file_queue, 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 = NomsuCompiler nomsu.arg = NomsuCompiler.list(args.nomsu_args) if args.version then nomsu:run([[use "core" say (Nomsu version)]]) os.exit(EXIT_SUCCESS) end 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 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 for _, filename in Files.walk(f) do input_files[filename] = true end end nomsu.can_optimize = function(f) if args.optimization == 0 then return false end if args.compile and input_files[f] then return false end return true end if not (args.no_core) then for _, filename in Files.walk('core') do if filename:match("%.nom$") then nomsu:run_file(filename) end end end local get_file_and_source get_file_and_source = function(filename) local file, source if filename == 'stdin' then file = io.read("*a") 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 return nil end source = Source(filename, 1, #file) return file, source end local run_file run_file = function(filename, lua_handler) if lua_handler == nil then lua_handler = nil end local file, source = get_file_and_source(filename) if not (file) then return end local tree = nomsu:parse(file, source) if tree then if tree.type ~= "FileChunks" then tree = { tree } end for _index_0 = 1, #tree do local chunk = tree[_index_0] local lua = nomsu:compile(chunk):as_statements("return ") lua:declare_locals() lua:prepend("-- File: " .. tostring(source.filename:gsub("\n.*", "...")) .. "\n") if lua_handler and input_files[filename] then lua_handler(tostring(lua)) end nomsu:run_lua(lua) end end end local parse_errs = { } for _index_0 = 1, #file_queue do local f = file_queue[_index_0] for _, filename in Files.walk(f) do local _continue_0 = false repeat if not (filename == "stdin" or filename:match("%.nom$")) then _continue_0 = true break end if args.check_syntax then local file, source = get_file_and_source(filename) if not (file) then _continue_0 = true break end nomsu:parse(file, source) print("Parse succeeded: " .. tostring(filename)) end if args.compile then local output if filename == 'stdin' then output = io.output() else output = io.open(filename:gsub("%.nom$", ".lua"), "w") end run_file(filename, function(lua) output:write(tostring(lua), "\n") if args.verbose then return print(tostring(lua)) end end) print(("Compiled %-25s -> %s"):format(filename, filename:gsub("%.nom$", ".lua"))) output:close() end if not args.check_syntax and not args.compile then run_file(filename, (args.verbose and print or nil)) end _continue_0 = true until true if not _continue_0 then break end end end if #file_queue == 0 then nomsu:run([[#!/usr/bin/env nomsu -V2 use "lib/consolecolor.nom" 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. say ".." \(bright)\(underscore)Welcome to the Nomsu v\(Nomsu version) interactive console!\(reset color) press 'enter' twice to run a command \("")]]) for repl_line = 1, math.huge do io.write(colored.bright(colored.yellow(">> "))) local buff = { } while true do io.write(colors.bright) local line = io.read("*L") io.write(colors.reset) if line == "\n" or not line then if #buff > 0 then io.write("\027[1A\027[2K") end break end line = line:gsub("\t", " ") table.insert(buff, line) io.write(colored.dim(colored.yellow(".. "))) end if #buff == 0 then break end buff = table.concat(buff) local pseudo_filename = "user input #" .. repl_line Files.spoof(pseudo_filename, buff) local err_hand err_hand = function(error_message) return Errhand.print_error(error_message) end local ret ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source(pseudo_filename, 1, #buff)) if ok and ret ~= nil then print("= " .. repr(ret)) elseif not ok then Errhand.print_error(ret) end end 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)