nomsu/nomsu.lua

349 lines
10 KiB
Lua
Raw Normal View History

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
2018-06-19 01:27:32 -07:00
local usage = [=[Nomsu Compiler
2018-09-12 15:31:59 -07:00
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 <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.
-e Execute the specified string.
2018-07-09 16:58:46 -07:00
-s Check the input files for syntax errors.
-I <file> Add an additional input file or directory.
2018-07-10 15:06:02 -07:00
-d <debugger> Attempt to use the specified debugger to wrap the main body of execution.
-h/--help Print this message.
2018-06-23 00:57:31 -07:00
--version Print the version number and exit.
--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
]=]
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")
2018-06-19 15:24:24 -07:00
local Errhand = require("error_handling")
2018-06-19 01:27:32 -07:00
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
2018-06-23 00:57:31 -07:00
local repr
repr = require("utils").repr
2018-06-19 01:27:32 -07:00
if not arg or debug.getinfo(2).func == require then
return NomsuCompiler
end
2018-07-09 16:58:46 -07:00
local file_queue = { }
2018-09-08 01:05:59 -07:00
local sep = "\3"
2018-09-12 15:31:59 -07:00
local parser = re.compile([[ args <- {| (flag %sep)* (({~ file ~} -> add_file) {:primary_file: %true :} %sep)?
{:nomsu_args: {| ({(!%sep .)*} %sep)* |} :} %sep? |} !.
2018-06-19 01:27:32 -07:00
flag <-
2018-09-12 15:31:59 -07:00
{:optimization: "-O" (%sep? %number)? :}
/ ("-I" %sep? ({~ file ~} -> add_file))
2018-09-12 15:31:59 -07:00
/ ("-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.])+})? :}
file <- ("-" -> "stdin") / {(!%sep .)+}
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,
sep = lpeg.P(sep),
2018-07-09 16:58:46 -07:00
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
2018-06-19 01:27:32 -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)
os.exit(EXIT_FAILURE)
2018-06-19 01:27:32 -07:00
end
local nomsu = NomsuCompiler
nomsu.arg = NomsuCompiler._List(args.nomsu_args)
2018-06-23 00:57:31 -07:00
if args.version then
nomsu:run([[use "core"
say (Nomsu version)]])
2018-06-23 00:57:31 -07:00
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
})
2018-06-19 01:27:32 -07:00
local run
run = function()
local input_files = { }
2018-07-09 16:58:46 -07:00
for _index_0 = 1, #file_queue do
2018-07-24 17:36:45 -07:00
local _continue_0 = false
repeat
local f = file_queue[_index_0]
if f == 'stdin' then
input_files[f] = true
_continue_0 = true
break
end
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
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
end
2018-06-23 00:57:31 -07:00
nomsu.can_optimize = function(f)
if args.optimization == 0 then
2018-06-23 00:57:31 -07:00
return false
end
2018-07-09 16:58:46 -07:00
if args.compile and input_files[f] then
2018-06-23 00:57:31 -07:00
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
2018-07-09 16:58:46 -07:00
local get_file_and_source
get_file_and_source = function(filename)
local file, source
2018-07-24 17:36:45 -07:00
if filename == 'stdin' or filename:match("%.nom$") then
file = Files.read(filename)
2018-07-09 16:58:46 -07:00
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
2018-07-09 16:58:46 -07:00
end
local tree = nomsu:parse(file, source)
if tree then
if tree.type ~= "FileChunks" then
2018-07-09 16:58:46 -07:00
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
2018-07-09 16:58:46 -07:00
lua_handler(tostring(lua))
end
nomsu:run_lua(lua)
end
end
end
2018-06-19 01:27:32 -07:00
local parse_errs = { }
2018-07-09 16:58:46 -07:00
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
2018-07-09 16:58:46 -07:00
if args.check_syntax then
local file, source = get_file_and_source(filename)
if not (file) then
_continue_0 = true
break
end
2018-09-12 15:31:59 -07:00
local tree = nomsu:parse(file, source)
2018-07-09 16:58:46 -07:00
print("Parse succeeded: " .. tostring(filename))
end
if args.compile then
2018-07-09 16:58:46 -07:00
local output
if filename == 'stdin' then
output = io.output()
else
2018-07-09 16:58:46 -07:00
output = io.open(filename:gsub("%.nom$", ".lua"), "w")
end
run_file(filename, function(lua)
output:write(tostring(lua), "\n")
if args.verbose then
2018-07-09 16:58:46 -07:00
return print(tostring(lua))
end
2018-07-09 16:58:46 -07:00
end)
print(("Compiled %-25s -> %s"):format(filename, filename:gsub("%.nom$", ".lua")))
output:close()
end
2018-07-09 16:58:46 -07:00
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
2018-06-19 01:27:32 -07:00
end
end
end
if not (args.primary_file or args.exec_strings) then
nomsu:run([[#!/usr/bin/env nomsu -V2
2018-06-23 18:26:18 -07:00
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.
2018-06-23 18:26:18 -07:00
say ".."
\(bright)\(underscore)Welcome to the Nomsu v\(Nomsu version) interactive console!\(reset color)
press 'enter' twice to run a command
2018-06-23 18:26:18 -07:00
\("")]])
2018-06-19 01:27:32 -07:00
for repl_line = 1, math.huge do
io.write(colored.bright(colored.yellow(">> ")))
local buff = { }
while true do
2018-06-23 00:57:31 -07:00
io.write(colors.bright)
2018-06-19 01:27:32 -07:00
local line = io.read("*L")
2018-06-23 00:57:31 -07:00
io.write(colors.reset)
2018-06-19 01:27:32 -07:00
if line == "\n" or not line then
if #buff > 0 then
io.write("\027[1A\027[2K")
end
2018-06-19 01:27:32 -07:00
break
end
2018-06-19 01:27:32 -07:00
line = line:gsub("\t", " ")
table.insert(buff, line)
io.write(colored.dim(colored.yellow(".. ")))
end
2018-06-19 01:27:32 -07:00
if #buff == 0 then
break
end
2018-06-19 01:27:32 -07:00
buff = table.concat(buff)
2018-06-19 15:24:24 -07:00
local pseudo_filename = "user input #" .. repl_line
Files.spoof(pseudo_filename, buff)
2018-06-19 01:27:32 -07:00
local err_hand
err_hand = function(error_message)
2018-06-19 15:24:24 -07:00
return Errhand.print_error(error_message)
end
local ret
ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source(pseudo_filename, 1, #buff))
2018-06-19 01:27:32 -07:00
if ok and ret ~= nil then
print("= " .. repr(ret))
elseif not ok then
2018-06-19 15:24:24 -07:00
Errhand.print_error(ret)
end
end
end
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)