Updated command line interface and compilation.

This commit is contained in:
Bruce Hill 2017-10-08 18:23:48 -07:00
parent 529afd465f
commit 7c231bf435
4 changed files with 173 additions and 103 deletions

2
.gitignore vendored Normal file
View File

@ -0,0 +1,2 @@
*.nom.lua
*/*.nom.lua

8
compile_lib.sh Executable file
View File

@ -0,0 +1,8 @@
#!/bin/sh
for file in $(find lib/ -name "*.nom") ; do
luafile="$file.lua"
if [ ! -e "$luafile" ] || [ "$file" -nt "$luafile" ] ; then
echo "Compiling $file into $luafile"
./nomsu.moon -c $file
fi
done

144
nomsu.lua
View File

@ -745,27 +745,42 @@ do
return lua, nil
end
self:defmacro("lua expr %code", lua_value)
local _require
_require = function(self, vars)
if not self.loaded_files[vars.filename] then
local run_file
run_file = function(self, vars)
if vars.filename:match(".*%.lua") then
return dofile(vars.filename)(self, vars)
end
if vars.filename:match(".*%.nom") then
if not self.skip_precompiled then
local file = io.open(vars.filename .. ".lua", "r")
if file then
local contents = file:read('*a')
file:close()
return load(contents)()(self, vars)
end
end
local file = io.open(vars.filename)
if not file then
self:error("File does not exist: " .. tostring(vars.filename))
end
self.loaded_files[vars.filename] = (self:run(file:read('*a'), vars.filename)) or true
local contents = file:read('*a')
file:close()
return self:run(contents, vars.filename)
else
return self:error("Invalid filetype for " .. tostring(vars.filename))
end
end
self:def("run file %filename", run_file)
local _require
_require = function(self, vars)
if not self.loaded_files[vars.filename] then
self.loaded_files[vars.filename] = run_file(self, {
filename = vars.filename
}) or true
end
return self.loaded_files[vars.filename]
end
self:def("require %filename", _require)
local run_file
run_file = function(self, vars)
local file = io.open(vars.filename)
if not file then
self:error("File does not exist: " .. tostring(vars.filename))
end
return self:run(file:read('*a'), vars.filename)
end
return self:def("run file %filename", run_file)
return self:def("require %filename", _require)
end
}
_base_0.__index = _base_0
@ -826,51 +841,74 @@ do
end
NomsuCompiler = _class_0
end
if arg and arg[1] then
local c = NomsuCompiler()
local input = io.open(arg[1]):read("*a")
local _write = c.write
if arg[2] == "-" then
c.write = function() end
if arg then
local parser = re.compile([[ args <- {| {:flags: flags? :} ({:input: input :} ";" ("-o;"{:output: output :} ";")?)? |} !.
flags <- (({| ({flag} ";")* |}) -> set)
flag <- "-c" / "-i" / "-p" / "-f" / "--help" / "-h"
input <- "-" / [^;]+
output <- "-" / [^;]+
]], {
set = utils.set
})
local args = concat(arg, ";") .. ";"
args = parser:match(args) or { }
if not args or not args.flags or args.flags["--help"] or args.flags["-h"] then
print("Usage: lua nomsu.lua [-c] [-i] [-p] [-f] [--help] [input [-o output]]")
os.exit()
end
local retval, code = c:run(input, arg[1])
c.write = _write
if arg[2] then
local output
if arg[2] == "-" then
output = io.output()
else
output = io.open(arg[2], 'w')
local c = NomsuCompiler()
c.skip_precompiled = args.flags["-f"]
if args.input then
if args.flags["-c"] and not args.output then
args.output = args.input .. ".lua"
end
local compiled_output = nil
if args.flags["-p"] then
local _write = c.write
c.write = function() end
compiled_output = io.output()
elseif args.output then
compiled_output = io.open(args.output, 'w')
end
if args.input:match(".*%.lua") then
local retval = dofile(args.input)(c, { })
else
local input
if args.input == '-' then
input = io.read('*a')
else
input = io.open(args.input):read("*a")
end
local retval, code = c:run(input, args.input)
if compiled_output then
compiled_output:write(code)
end
end
if args.flags["-p"] then
c.write = _write
end
output:write(([[ local NomsuCompiler = require('nomsu');
local c = NomsuCompiler();
local run = (function(nomsu, vars)
%s
end);
return run(c, {});
]]):format(code))
end
elseif arg then
local c = NomsuCompiler()
c:run('require "lib/core.nom"')
while true do
local buff = ""
if not args.input or args.flags["-i"] then
c:run('require "lib/core.nom"')
while true do
io.write(">> ")
local line = io.read("*L")
if line == "\n" or not line then
local buff = ""
while true do
io.write(">> ")
local line = io.read("*L")
if line == "\n" or not line then
break
end
buff = buff .. line
end
if #buff == 0 then
break
end
buff = buff .. line
end
if #buff == 0 then
break
end
local ok, ret = pcall(function()
return c:run(buff)
end)
if ok and ret ~= nil then
print("= " .. repr(ret))
local ok, ret = pcall(function()
return c:run(buff)
end)
if ok and ret ~= nil then
print("= " .. repr(ret))
end
end
end
end

View File

@ -21,7 +21,6 @@ colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..msg..col
--pcall = (fn,...)-> true, fn(...)
-- TODO:
-- check robustness/functionality of compiler mode.
-- Maybe get GOTOs working at file scope.
-- use actual variables instead of a vars table
-- consider non-linear codegen, rather than doing thunks for things like comprehensions
@ -564,66 +563,89 @@ class NomsuCompiler
return lua, nil
@defmacro "lua expr %code", lua_value
_require = (vars)=>
if not @loaded_files[vars.filename]
run_file = (vars)=>
if vars.filename\match(".*%.lua")
return dofile(vars.filename)(@, vars)
if vars.filename\match(".*%.nom")
if not @skip_precompiled -- Look for precompiled version
file = io.open(vars.filename..".lua", "r")
if file
contents = file\read('*a')
file\close!
return load(contents)!(@, vars)
file = io.open(vars.filename)
if not file
@error "File does not exist: #{vars.filename}"
@loaded_files[vars.filename] = (@run(file\read('*a'), vars.filename)) or true
contents = file\read('*a')
file\close!
return @run(contents, vars.filename)
else
@error "Invalid filetype for #{vars.filename}"
@def "run file %filename", run_file
_require = (vars)=>
if not @loaded_files[vars.filename]
@loaded_files[vars.filename] = run_file(self, {filename:vars.filename}) or true
return @loaded_files[vars.filename]
@def "require %filename", _require
run_file = (vars)=>
file = io.open(vars.filename)
if not file
@error "File does not exist: #{vars.filename}"
return @run(file\read('*a'), vars.filename)
@def "run file %filename", run_file
if arg
parser = re.compile([[
args <- {| {:flags: flags? :} ({:input: input :} ";" ("-o;"{:output: output :} ";")?)? |} !.
flags <- (({| ({flag} ";")* |}) -> set)
flag <- "-c" / "-i" / "-p" / "-f" / "--help" / "-h"
input <- "-" / [^;]+
output <- "-" / [^;]+
]], {set: utils.set})
args = concat(arg, ";")..";"
args = parser\match(args) or {}
if not args or not args.flags or args.flags["--help"] or args.flags["-h"]
print "Usage: lua nomsu.lua [-c] [-i] [-p] [-f] [--help] [input [-o output]]"
os.exit!
if arg and arg[1]
--ProFi = require 'ProFi'
--ProFi\start()
c = NomsuCompiler()
input = io.open(arg[1])\read("*a")
-- If run via "./nomsu.moon file.nom -", then silence output and print generated
-- source code instead.
_write = c.write
if arg[2] == "-"
c.write = ->
retval, code = c\run(input, arg[1])
c.write = _write -- put it back
if arg[2]
output = if arg[2] == "-"
io.output()
else io.open(arg[2], 'w')
output\write ([[
local NomsuCompiler = require('nomsu');
local c = NomsuCompiler();
local run = (function(nomsu, vars)
%s
end);
return run(c, {});
]])\format(code)
--ProFi\stop()
--ProFi\writeReport( 'MyProfilingReport.txt' )
c.skip_precompiled = args.flags["-f"]
if args.input
-- Read a file or stdin and output either the printouts or the compiled lua
if args.flags["-c"] and not args.output
args.output = args.input..".lua"
compiled_output = nil
if args.flags["-p"]
_write = c.write
c.write = ->
compiled_output = io.output()
elseif args.output
compiled_output = io.open(args.output, 'w')
elseif arg
-- REPL:
c = NomsuCompiler()
c\run('require "lib/core.nom"')
while true
buff = ""
if args.input\match(".*%.lua")
retval = dofile(args.input)(c, {})
else
input = if args.input == '-'
io.read('*a')
else io.open(args.input)\read("*a")
retval, code = c\run(input, args.input)
-- Output compile lua code
if compiled_output
compiled_output\write code
if args.flags["-p"]
c.write = _write
if not args.input or args.flags["-i"]
-- REPL
c\run('require "lib/core.nom"')
while true
io.write(">> ")
line = io.read("*L")
if line == "\n" or not line
buff = ""
while true
io.write(">> ")
line = io.read("*L")
if line == "\n" or not line
break
buff ..= line
if #buff == 0
break
buff ..= line
if #buff == 0
break
ok, ret = pcall(-> c\run(buff))
if ok and ret != nil
print "= "..repr(ret)
ok, ret = pcall(-> c\run(buff))
if ok and ret != nil
print "= "..repr(ret)
return NomsuCompiler