nomsu/nomsu.lua

346 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
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
2018-09-16 16:57:14 -07:00
nomsu.environment.arg = NomsuCompiler.environment._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
nomsu:import_file('core')
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
2018-09-14 19:17:09 -07:00
nomsu:run([[#!/usr/bin/env nomsu -V4
2018-06-23 18:26:18 -07:00
use "lib/consolecolor.nom"
[quit, exit] all mean: lua> "os.exit(0)"
(help) means:
2018-09-14 19:17:09 -07:00
say "\
..This is the Nomsu v\(Nomsu version) interactive console.
You can type in Nomsu code here and hit 'enter' twice to run it.
2018-09-14 19:17:09 -07:00
To exit, type 'exit' or 'quit' and hit enter twice."
2018-06-23 18:26:18 -07:00
2018-09-14 19:17:09 -07:00
say "\
..
2018-06-23 18:26:18 -07:00
\(bright)\(underscore)Welcome to the Nomsu v\(Nomsu version) interactive console!\(reset color)
press 'enter' twice to run a command
2018-09-14 19:17:09 -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, NomsuCode(Source(pseudo_filename, 1, #buff), buff))
2018-06-19 01:27:32 -07:00
if ok and ret ~= nil then
if type(ret) == 'number' then
print("= " .. tostring(ret))
else
print("= " .. tostring(ret:as_nomsu()))
end
2018-06-19 01:27:32 -07:00
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)