local EXIT_SUCCESS, EXIT_FAILURE = 0, 1 local usage = [=[Nomsu Compiler Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-v] [-c] [-f] [-s] [--help] [-p print_file] file1 file2... [-- nomsu args...] 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) -v Verbose: print compiled lua code -c Compile .nom files into .lua files -f Auto-format the given Nomsu file and print the result. -s Check the program for syntax errors. -h/--help Print this message. -p Print to the specified file instead of stdout. Input file 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 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 STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2" 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) :} file <- "-" / [^;]+ ]], { ["true"] = function() return true end }) local args = table.concat(arg, ";") .. ";" args = parser:match(args) if not args or args.help then print(usage) os.exit(EXIT_FAILURE) end local nomsu = NomsuCompiler nomsu.arg = args.nomsu_args 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 match, sub, rep, gsub, format, byte, find do local _obj_0 = string match, sub, rep, gsub, format, byte, match, find = _obj_0.match, _obj_0.sub, _obj_0.rep, _obj_0.gsub, _obj_0.format, _obj_0.byte, _obj_0.match, _obj_0.find end local iterate_single iterate_single = function(item, prev) if item == prev then return nil else return item end end local lfs ok, lfs = pcall(require, "lfs") if ok then all_files = function(path) local browse browse = function(filename) local file_type = lfs.attributes(filename, 'mode') if file_type == 'file' then if match(filename, "%.nom$") or match(filename, "%.lua$") then coroutine.yield(filename) return true end elseif file_type == 'directory' then for subfile in lfs.dir(filename) do if not (subfile == "." or subfile == "..") then browse(filename .. "/" .. subfile) end end return true elseif file_type == 'char device' then coroutine.yield(filename) return true end return false end return coroutine.wrap(function() if not browse(path) and package.nomsupath then browse(package.nomsupath .. "/" .. path) end return nil end) end else local ret = os.execute('find . -maxdepth 0') if not (ret == true or ret == 0) then error("Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: http://keplerproject.github.io/luafilesystem/ or `luarocks install luafilesystem`)", 0) end all_files = function(path) if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") then return iterate_single, path end path = gsub(path, "\\", "\\\\") path = gsub(path, "`", "") path = gsub(path, '"', '\\"') path = gsub(path, "$", "") return coroutine.wrap(function() local f = io.popen('find -L "' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"') local found = false for line in f:lines() do found = true coroutine.yield(line) end if not found and package.nomsupath then f:close() f = io.popen('find -L "' .. package.nomsupath .. '/' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"') for line in f:lines() do coroutine.yield(line) end end local success = f:close() if not (success) then return error("Invalid file path: " .. tostring(path)) end end) end end local run run = function() for i, input in ipairs(args.inputs) do if input == "-" then args.inputs[i] = STDIN 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 local input = _list_0[_index_0] for f in all_files(input) do input_files[#input_files + 1] = f to_run[f] = true end end nomsu.skip_precompiled = to_run if args.compile or args.verbose then nomsu.on_compile = function(code, from_file) if not (to_run[from_file]) then return end if args.verbose then io.write(tostring(code), "\n") end if args.compile and from_file:match("%.nom$") then local output_filename = from_file:gsub("%.nom$", ".lua") local output_file = io.open(output_filename, 'w') output_file:write(tostring(code)) output_file:flush() print(("Compiled %-25s -> %s"):format(from_file, output_filename)) return output_file:close() end end end local parse_errs = { } for _index_0 = 1, #input_files do local filename = input_files[_index_0] if args.syntax then local file_contents = io.open(filename):read('*a') local err ok, err = pcall(nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents)) 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 elseif args.format then local file = FILE_CACHE[filename] if not file then error("File does not exist: " .. tostring(filename), 0) end local tree = nomsu:parse(file, Source(filename, 1, #file)) local formatted = tostring(nomsu:tree_to_nomsu(tree)) if print_file then print_file:write(formatted, "\n") print_file:flush() end elseif filename == STDIN then local file = io.input():read("*a") FILE_CACHE.stdin = file nomsu:run(file, Source('stdin', 1, #file)) else nomsu:run_file(filename) end end if #parse_errs > 0 then io.stderr:write(table.concat(parse_errs, "\n\n")) io.stderr:flush() os.exit(EXIT_FAILURE) elseif args.syntax then os.exit(EXIT_SUCCESS) end if args.interactive then for repl_line = 1, math.huge do io.write(colored.bright(colored.yellow(">> "))) local buff = { } while true do local line = io.read("*L") 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 FILE_CACHE[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 has_ldt, ldt = pcall(require, 'ldt') if has_ldt then return ldt.guard(run) else return Errhand.run_safely(run) end