Cleaned up/improved the command line interface, and fixed the
REPL/autoformatting.
This commit is contained in:
parent
5629b81f78
commit
9d6932a872
@ -11,8 +11,8 @@ Nomsu's only dependencies are [Lua 5.2 or later](https://www.lua.org/) (tested w
|
||||
## Usage
|
||||
|
||||
* To get a nomsu [REPL](https://en.wikipedia.org/wiki/Read-eval-print_loop), simply run `lua nomsu.lua`.
|
||||
* To run a .nom file with nomsu code, run `lua nomsu.lua your_file.nom`. Or `lua nomsu.lua -` if reading from stdin.
|
||||
* (Advanced/optional) To precompile a .nom file into lua, run `lua nomsu.lua -o <output_filename> <input_file>` to precompile your file into lua (by default, saved at `<your_file>.lua`). If you run the compiler with the "-O" (for Optimized) flag, the compiler will run the precompiled lua files directly, instead of parsing and compiling the .nom files before running them. This is not necessary, but it can speed things up if you precompile a bunch of code that involves macros or compile-time activity.
|
||||
* To run a .nom file with nomsu code, run `lua nomsu.lua your_file.nom`.
|
||||
* (Advanced/optional) To precompile a .nom file into lua, run `lua nomsu.lua -c <input_file>` to precompile your file into lua (by default, saved in the same location, but with a `.lua` extension instead of `.nom`). If you run the compiler with the "-O" (for Optimized) flag, the compiler will run the precompiled lua files directly, instead of parsing and compiling the .nom files before running them. This is not necessary, but it can speed things up if you precompile a bunch of code that involves macros or compile-time activity.
|
||||
* More usage options are avilable via `lua nomsu.lua --help`.
|
||||
|
||||
## Layout
|
||||
|
@ -4,13 +4,5 @@
|
||||
set -e
|
||||
moonc *.moon
|
||||
rm -f core/*.lua lib/*.lua
|
||||
for file in core/*.nom; do
|
||||
printf "Compiling $file ..."
|
||||
luajit ./nomsu.lua -O -o "core/$(basename $file .nom).lua" $file
|
||||
echo "done."
|
||||
done
|
||||
for file in lib/*.nom; do
|
||||
printf "Compiling $file ..."
|
||||
luajit ./nomsu.lua -O -o "lib/$(basename $file .nom).lua" $file
|
||||
echo "done."
|
||||
done
|
||||
luajit ./nomsu.lua -c core lib
|
||||
echo "done."
|
||||
|
128
nomsu.lua
128
nomsu.lua
@ -316,10 +316,7 @@ do
|
||||
end
|
||||
return tree
|
||||
end,
|
||||
run = function(self, nomsu_code, compile_fn)
|
||||
if compile_fn == nil then
|
||||
compile_fn = nil
|
||||
end
|
||||
run = function(self, nomsu_code)
|
||||
local tree = assert(self:parse(nomsu_code))
|
||||
if type(tree) == 'number' then
|
||||
return nil
|
||||
@ -327,15 +324,12 @@ do
|
||||
local lua = self:tree_to_lua(tree):as_statements()
|
||||
lua:declare_locals()
|
||||
lua:prepend("-- File: " .. tostring(nomsu_code.source or "") .. "\n")
|
||||
if compile_fn then
|
||||
compile_fn(lua)
|
||||
if self.compile_fn then
|
||||
self.compile_fn(lua, nomsu_code.source.filename)
|
||||
end
|
||||
return self:run_lua(lua)
|
||||
end,
|
||||
run_file = function(self, filename, compile_fn)
|
||||
if compile_fn == nil then
|
||||
compile_fn = nil
|
||||
end
|
||||
run_file = function(self, filename)
|
||||
local loaded = self.environment.LOADED
|
||||
if loaded[filename] then
|
||||
return loaded[filename]
|
||||
@ -384,7 +378,7 @@ do
|
||||
if not file then
|
||||
error("File does not exist: " .. tostring(filename), 0)
|
||||
end
|
||||
ret = self:run(Nomsu(Source(filename, 1, #file), file), compile_fn)
|
||||
ret = self:run(Nomsu(Source(filename, 1, #file), file))
|
||||
else
|
||||
error("Invalid filetype for " .. tostring(filename), 0)
|
||||
end
|
||||
@ -1355,7 +1349,8 @@ if arg and debug_getinfo(2).func ~= require then
|
||||
/ {:format: ("-f" -> true) :}
|
||||
/ {:syntax: ("-s" -> true) :}
|
||||
/ {:print_file: "-p" ";" {file} :}
|
||||
/ {:output_file: "-o" ";" {file} :}
|
||||
/ {:compile: ("-c" -> true) :}
|
||||
/ {:verbose: ("-v" -> true) :}
|
||||
/ {:help: (("-h" / "--help") -> true) :}
|
||||
file <- "-" / [^;]+
|
||||
]], {
|
||||
@ -1368,15 +1363,16 @@ if arg and debug_getinfo(2).func ~= require then
|
||||
if not args or args.help then
|
||||
print([=[Nomsu Compiler
|
||||
|
||||
Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-f] [-s] [--help] [-o output] [-p print_file] file1 file2... [-- nomsu args...]
|
||||
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.
|
||||
-o <file> Output the compiled Lua file to the given file (use "-" to output to stdout; if outputting to stdout and -p is not specified, -p will default to /dev/null)
|
||||
-p <file> Print to the specified file instead of stdout.
|
||||
<input> Input file can be "-" to use stdin.
|
||||
]=])
|
||||
@ -1613,8 +1609,6 @@ OPTIONS
|
||||
print_file = io.stdout
|
||||
elseif args.print_file then
|
||||
print_file = io.open(args.print_file, 'w')
|
||||
elseif args.output_file == '-' then
|
||||
print_file = nil
|
||||
else
|
||||
print_file = io.stdout
|
||||
end
|
||||
@ -1634,55 +1628,64 @@ OPTIONS
|
||||
return print_file:flush()
|
||||
end
|
||||
end
|
||||
local compile_fn
|
||||
if args.output_file then
|
||||
compile_fn = function(code)
|
||||
local output_file
|
||||
if args.output_file == "-" then
|
||||
output_file = io.stdout
|
||||
elseif args.output_file then
|
||||
output_file = io.open(args.output_file, 'w')
|
||||
end
|
||||
output_file:write("local IMMEDIATE = true;\n" .. tostring(code))
|
||||
return output_file:flush()
|
||||
end
|
||||
else
|
||||
compile_fn = nil
|
||||
end
|
||||
local parse_errs = { }
|
||||
local input_files = { }
|
||||
local to_run = { }
|
||||
local _list_0 = args.inputs
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local input = _list_0[_index_0]
|
||||
if args.syntax then
|
||||
for input_file in all_files(input) do
|
||||
local err
|
||||
ok, err = pcall(nomsu.parse, nomsu, Nomsu(input_file, io.open(input_file):read("*a")))
|
||||
if not ok then
|
||||
insert(parse_errs, err)
|
||||
elseif print_file then
|
||||
print_file:write("Parse succeeded: " .. tostring(input_file) .. "\n")
|
||||
print_file:flush()
|
||||
for f in all_files(input) do
|
||||
input_files[#input_files + 1] = f
|
||||
to_run[f] = true
|
||||
end
|
||||
end
|
||||
if args.compile or args.verbose then
|
||||
nomsu.compile_fn = function(code, from_file)
|
||||
if to_run[from_file] then
|
||||
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("local IMMEDIATE = true;\n", tostring(code))
|
||||
output_file:flush()
|
||||
print(("Compiled %-25s -> %s"):format(from_file, output_filename))
|
||||
return output_file:close()
|
||||
end
|
||||
end
|
||||
end
|
||||
else
|
||||
nomsu.compile_fn = nil
|
||||
end
|
||||
local parse_errs = { }
|
||||
for _index_0 = 1, #input_files do
|
||||
local filename = input_files[_index_0]
|
||||
if args.syntax then
|
||||
local err
|
||||
ok, err = pcall(nomsu.parse, nomsu, Nomsu(filename, io.open(filename):read("*a")))
|
||||
if not ok then
|
||||
insert(parse_errs, err)
|
||||
elseif print_file then
|
||||
print_file:write("Parse succeeded: " .. tostring(filename) .. "\n")
|
||||
print_file:flush()
|
||||
end
|
||||
elseif args.format then
|
||||
for input_file in all_files(input) do
|
||||
local tree = nomsu:parse(io.open(input_file):read("*a"))
|
||||
local formatted = tostring(self:tree_to_nomsu(tree))
|
||||
if output_file then
|
||||
output_file:write(formatted, "\n")
|
||||
output_file:flush()
|
||||
end
|
||||
if print_file then
|
||||
print_file:write(formatted, "\n")
|
||||
print_file:flush()
|
||||
end
|
||||
local file = FILE_CACHE[filename]
|
||||
if not file then
|
||||
error("File does not exist: " .. tostring(filename), 0)
|
||||
end
|
||||
elseif input == STDIN then
|
||||
local tree = nomsu:parse(Nomsu(Source(filename, 1, #file), 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(Nomsu(Source('stdin', 1, #file), file), compile_fn)
|
||||
nomsu:run(Nomsu(Source('stdin', 1, #file), file))
|
||||
else
|
||||
nomsu:run_file(input, compile_fn)
|
||||
nomsu:run_file(filename)
|
||||
end
|
||||
end
|
||||
if #parse_errs > 0 then
|
||||
@ -1693,9 +1696,9 @@ OPTIONS
|
||||
os.exit(true, true)
|
||||
end
|
||||
if args.interactive then
|
||||
while true do
|
||||
for repl_line = 1, math.huge do
|
||||
io.write(colored.bright(colored.yellow(">> ")))
|
||||
local buff = ""
|
||||
local buff = { }
|
||||
while true do
|
||||
local line = io.read("*L")
|
||||
if line == "\n" or not line then
|
||||
@ -1705,14 +1708,21 @@ OPTIONS
|
||||
break
|
||||
end
|
||||
line = line:gsub("\t", " ")
|
||||
buff = buff .. line
|
||||
insert(buff, line)
|
||||
io.write(colored.dim(colored.yellow(".. ")))
|
||||
end
|
||||
if #buff == 0 then
|
||||
break
|
||||
end
|
||||
buff = concat(buff)
|
||||
FILE_CACHE["REPL#" .. repl_line] = buff
|
||||
local code = Nomsu(Source("REPL#" .. repl_line, 1, #buff), buff)
|
||||
local err_hand
|
||||
err_hand = function(error_message)
|
||||
return print_err_msg(error_message)
|
||||
end
|
||||
local ret
|
||||
ok, ret = pcall(nomsu.run, nomsu, buff)
|
||||
ok, ret = xpcall(nomsu.run, err_hand, nomsu, code)
|
||||
if ok and ret ~= nil then
|
||||
print("= " .. repr(ret))
|
||||
elseif not ok then
|
||||
|
98
nomsu.moon
98
nomsu.moon
@ -9,7 +9,7 @@
|
||||
-- nomsu = Nomsu()
|
||||
-- nomsu:run(your_nomsu_code)
|
||||
-- Or from the command line:
|
||||
-- lua nomsu.lua [input_file [output_file or -]]
|
||||
-- lua nomsu.lua your_file.nom
|
||||
export lpeg, re
|
||||
_pairs, _ipairs = pairs, ipairs
|
||||
if jit
|
||||
@ -324,19 +324,19 @@ class NomsuCompiler
|
||||
|
||||
return tree
|
||||
|
||||
run: (nomsu_code, compile_fn=nil)=>
|
||||
run: (nomsu_code)=>
|
||||
tree = assert(@parse(nomsu_code))
|
||||
if type(tree) == 'number' -- Happens if pattern matches, but there are no captures, e.g. an empty string
|
||||
return nil
|
||||
lua = @tree_to_lua(tree)\as_statements!
|
||||
lua\declare_locals!
|
||||
lua\prepend "-- File: #{nomsu_code.source or ""}\n"
|
||||
if compile_fn
|
||||
compile_fn(lua)
|
||||
if @compile_fn
|
||||
self.compile_fn(lua, nomsu_code.source.filename)
|
||||
return @run_lua(lua)
|
||||
|
||||
_running_files = {} -- For detecting circular imports
|
||||
run_file: (filename, compile_fn=nil)=>
|
||||
run_file: (filename)=>
|
||||
loaded = @environment.LOADED
|
||||
if loaded[filename]
|
||||
return loaded[filename]
|
||||
@ -367,7 +367,7 @@ class NomsuCompiler
|
||||
file = file or FILE_CACHE[filename]
|
||||
if not file
|
||||
error("File does not exist: #{filename}", 0)
|
||||
ret = @run(Nomsu(Source(filename,1,#file), file), compile_fn)
|
||||
ret = @run(Nomsu(Source(filename,1,#file), file))
|
||||
else
|
||||
error("Invalid filetype for #{filename}", 0)
|
||||
loaded[filename] = ret or true
|
||||
@ -966,7 +966,8 @@ if arg and debug_getinfo(2).func != require
|
||||
/ {:format: ("-f" -> true) :}
|
||||
/ {:syntax: ("-s" -> true) :}
|
||||
/ {:print_file: "-p" ";" {file} :}
|
||||
/ {:output_file: "-o" ";" {file} :}
|
||||
/ {:compile: ("-c" -> true) :}
|
||||
/ {:verbose: ("-v" -> true) :}
|
||||
/ {:help: (("-h" / "--help") -> true) :}
|
||||
file <- "-" / [^;]+
|
||||
]], {true: -> true})
|
||||
@ -976,15 +977,16 @@ if arg and debug_getinfo(2).func != require
|
||||
print [=[
|
||||
Nomsu Compiler
|
||||
|
||||
Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-f] [-s] [--help] [-o output] [-p print_file] file1 file2... [-- nomsu args...]
|
||||
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.
|
||||
-o <file> Output the compiled Lua file to the given file (use "-" to output to stdout; if outputting to stdout and -p is not specified, -p will default to /dev/null)
|
||||
-p <file> Print to the specified file instead of stdout.
|
||||
<input> Input file can be "-" to use stdin.
|
||||
]=]
|
||||
@ -1131,7 +1133,6 @@ OPTIONS
|
||||
|
||||
print_file = if args.print_file == "-" then io.stdout
|
||||
elseif args.print_file then io.open(args.print_file, 'w')
|
||||
elseif args.output_file == '-' then nil
|
||||
else io.stdout
|
||||
|
||||
nomsu.skip_precompiled = not args.optimized
|
||||
@ -1147,42 +1148,53 @@ OPTIONS
|
||||
print_file\write('\n')
|
||||
print_file\flush!
|
||||
|
||||
compile_fn = if args.output_file
|
||||
(code)->
|
||||
output_file = if args.output_file == "-" then io.stdout
|
||||
elseif args.output_file then io.open(args.output_file, 'w')
|
||||
output_file\write("local IMMEDIATE = true;\n"..tostring(code))
|
||||
output_file\flush!
|
||||
input_files = {}
|
||||
to_run = {}
|
||||
for input in *args.inputs
|
||||
for f in all_files(input)
|
||||
input_files[#input_files+1] = f
|
||||
to_run[f] = true
|
||||
|
||||
nomsu.compile_fn = if args.compile or args.verbose
|
||||
(code, from_file)->
|
||||
if to_run[from_file]
|
||||
if args.verbose
|
||||
io.write(tostring(code), "\n")
|
||||
if args.compile and from_file\match("%.nom$")
|
||||
output_filename = from_file\gsub("%.nom$", ".lua")
|
||||
output_file = io.open(output_filename, 'w')
|
||||
output_file\write("local IMMEDIATE = true;\n", tostring(code))
|
||||
output_file\flush!
|
||||
print ("Compiled %-25s -> %s")\format(from_file, output_filename)
|
||||
output_file\close!
|
||||
else nil
|
||||
|
||||
parse_errs = {}
|
||||
for input in *args.inputs
|
||||
for filename in *input_files
|
||||
if args.syntax
|
||||
-- Check syntax:
|
||||
for input_file in all_files(input)
|
||||
ok,err = pcall nomsu.parse, nomsu, Nomsu(input_file, io.open(input_file)\read("*a"))
|
||||
if not ok
|
||||
insert parse_errs, err
|
||||
elseif print_file
|
||||
print_file\write("Parse succeeded: #{input_file}\n")
|
||||
print_file\flush!
|
||||
ok,err = pcall nomsu.parse, nomsu, Nomsu(filename, io.open(filename)\read("*a"))
|
||||
if not ok
|
||||
insert parse_errs, err
|
||||
elseif print_file
|
||||
print_file\write("Parse succeeded: #{filename}\n")
|
||||
print_file\flush!
|
||||
elseif args.format
|
||||
-- Auto-format
|
||||
for input_file in all_files(input)
|
||||
tree = nomsu\parse(io.open(input_file)\read("*a"))
|
||||
formatted = tostring(@tree_to_nomsu(tree))
|
||||
if output_file
|
||||
output_file\write(formatted, "\n")
|
||||
output_file\flush!
|
||||
if print_file
|
||||
print_file\write(formatted, "\n")
|
||||
print_file\flush!
|
||||
elseif input == STDIN
|
||||
file = FILE_CACHE[filename]
|
||||
if not file
|
||||
error("File does not exist: #{filename}", 0)
|
||||
tree = nomsu\parse(Nomsu(Source(filename,1,#file), file))
|
||||
formatted = tostring(nomsu\tree_to_nomsu(tree))
|
||||
if print_file
|
||||
print_file\write(formatted, "\n")
|
||||
print_file\flush!
|
||||
elseif filename == STDIN
|
||||
file = io.input!\read("*a")
|
||||
FILE_CACHE.stdin = file
|
||||
nomsu\run(Nomsu(Source('stdin',1,#file), file), compile_fn)
|
||||
nomsu\run(Nomsu(Source('stdin',1,#file), file))
|
||||
else
|
||||
nomsu\run_file(input, compile_fn)
|
||||
nomsu\run_file(filename)
|
||||
|
||||
if #parse_errs > 0
|
||||
io.stderr\write concat(parse_errs, "\n\n")
|
||||
@ -1193,9 +1205,9 @@ OPTIONS
|
||||
|
||||
if args.interactive
|
||||
-- REPL
|
||||
while true
|
||||
for repl_line=1,math.huge
|
||||
io.write(colored.bright colored.yellow ">> ")
|
||||
buff = ""
|
||||
buff = {}
|
||||
while true
|
||||
line = io.read("*L")
|
||||
if line == "\n" or not line
|
||||
@ -1203,11 +1215,17 @@ OPTIONS
|
||||
io.write("\027[1A\027[2K")
|
||||
break -- Run buffer
|
||||
line = line\gsub("\t", " ")
|
||||
buff ..= line
|
||||
insert buff, line
|
||||
io.write(colored.dim colored.yellow ".. ")
|
||||
if #buff == 0
|
||||
break -- Exit
|
||||
ok, ret = pcall(nomsu.run, nomsu, buff)
|
||||
|
||||
buff = concat(buff)
|
||||
FILE_CACHE["REPL#"..repl_line] = buff
|
||||
code = Nomsu(Source("REPL#"..repl_line, 1, #buff), buff)
|
||||
err_hand = (error_message)->
|
||||
print_err_msg error_message
|
||||
ok, ret = xpcall(nomsu.run, err_hand, nomsu, code)
|
||||
if ok and ret != nil
|
||||
print "= "..repr(ret)
|
||||
elseif not ok
|
||||
|
Loading…
Reference in New Issue
Block a user