nomsu/nomsu.moon

253 lines
9.0 KiB
Plaintext
Raw Normal View History

2017-09-05 23:51:35 -07:00
#!/usr/bin/env moon
2018-06-19 01:27:32 -07:00
-- This file contains the command-line Nomsu runner.
if NOMSU_VERSION and NOMSU_PREFIX
ver_bits = [ver_bit for ver_bit in NOMSU_VERSION\gmatch("[0-9]+")]
partial_vers = [table.concat(ver_bits,'.',1,i) for i=#ver_bits,1,-1]
package.path = table.concat(["#{NOMSU_PREFIX}/share/nomsu/#{v}/?.lua" for v in *partial_vers],";")..";"..package.path
package.cpath = table.concat(["#{NOMSU_PREFIX}/lib/nomsu/#{v}/?.so" for v in *partial_vers],";")..";"..package.cpath
package.nomsupath = table.concat(["#{NOMSU_PREFIX}/share/nomsu/#{v}" for v in *partial_vers],";")
EXIT_SUCCESS, EXIT_FAILURE = 0, 1
2018-06-19 01:27:32 -07:00
usage = [=[
Nomsu Compiler
2018-06-23 00:57:31 -07:00
Usage: (lua nomsu.lua | moon nomsu.moon) [-V version] [-i] [-O] [-v] [-c] [-f] [-s] [--help] [--version] [-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.
2018-06-23 00:57:31 -07:00
--version Print the version number and exit.
-V specify which Nomsu version is desired
-p <file> Print to the specified file instead of stdout.
2018-04-28 19:16:39 -07:00
<input> Input file can be "-" to use stdin.
]=]
2018-04-28 19:16:39 -07:00
ok, _ = pcall ->
export lpeg, re
lpeg = require 'lpeg'
re = require 're'
if not ok
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)
2018-06-19 15:24:24 -07:00
Errhand = require "error_handling"
2018-06-19 01:27:32 -07:00
NomsuCompiler = require "nomsu_compiler"
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
2018-06-23 00:57:31 -07:00
{:repr} = require "utils"
2018-06-19 01:27:32 -07:00
STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2"
2018-06-19 01:27:32 -07:00
-- If this file was reached via require(), then just return the Nomsu compiler
if not arg or debug.getinfo(2).func == require
return NomsuCompiler
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) :}
2018-06-23 00:57:31 -07:00
/ {:version: ("--version" -> true) :}
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
2018-06-19 01:27:32 -07:00
file <- "-" / [^;]+
]], {true: -> true})
2018-06-23 00:57:31 -07:00
arg_string = table.concat(arg, ";")..";"
args = parser\match(arg_string)
2018-06-19 01:27:32 -07:00
if not args or args.help
print usage
os.exit(EXIT_FAILURE)
2018-06-19 01:27:32 -07:00
files = require "files"
2018-06-19 01:27:32 -07:00
nomsu = NomsuCompiler
nomsu.arg = args.nomsu_args
2018-06-23 00:57:31 -07:00
if args.version
nomsu\run [[
use "core"
say (Nomsu version)]]
2018-06-23 00:57:31 -07:00
os.exit(EXIT_SUCCESS)
export FILE_CACHE
-- FILE_CACHE is a map from filename (string) -> string of file contents
FILE_CACHE = setmetatable {}, {
__index: (filename)=>
file = io.open(filename)
return nil unless file
contents = file\read("*a")
file\close!
self[filename] = contents
return contents
}
2018-06-19 01:27:32 -07:00
run = ->
for i,input in ipairs args.inputs
if input == "-" then args.inputs[i] = STDIN
if #args.inputs == 0 and not args.interactive
args.inputs = {"core"}
args.interactive = true
print_file = if args.print_file == "-" then io.stdout
elseif args.print_file then io.open(args.print_file, 'w')
else io.stdout
if print_file == nil
nomsu.print = ->
elseif print_file != io.stdout
nomsu.print = (...)->
N = select("#",...)
if N > 0
print_file\write(tostring(select(1,...)))
for i=2,N
print_file\write('\t',tostring(select(1,...)))
print_file\write('\n')
print_file\flush!
input_files = {}
to_run = {}
for input in *args.inputs
found = false
for f in files.walk(input)
2018-06-19 01:27:32 -07:00
input_files[#input_files+1] = f
to_run[f] = true
found = true
if not found
error("Could not find: #{input}")
2018-06-19 01:27:32 -07:00
2018-06-23 00:57:31 -07:00
nomsu.can_optimize = (f)->
return false unless args.optimized
return false if to_run[f]
return true
2018-06-19 01:27:32 -07:00
parse_errs = {}
for arg in *args.inputs
for filename in files.walk(arg)
local file, source
if filename == STDIN
file = io.input!\read("*a")
files.spoof('stdin', file)
source = Source('stdin', 1, #file)
elseif filename\match("%.nom$")
file = files.read(filename)
if not file
error("File does not exist: #{filename}", 0)
source = Source(filename, 1, #file)
else continue
source = Source(filename,1,#file)
output = if args.compile then io.open(filename\gsub("%.nom$", ".lua"), "w") else nil
if args.syntax
-- Check syntax:
ok,err = pcall nomsu.parse, nomsu, file, source
if not ok
table.insert parse_errs, err
elseif print_file
print_file\write("Parse succeeded: #{filename}\n")
print_file\flush!
continue
tree = nomsu\parse(file, source)
if args.format
-- Auto-format
formatted = tree and tostring(nomsu\tree_to_nomsu(tree)) or ""
if print_file
print_file\write(formatted, "\n")
print_file\flush!
continue
if tree
if tree.type == "FileChunks"
-- Each chunk's compilation is affected by the code in the previous chunks
-- (typically), so each chunk needs to compile and run before the next one
-- compiles.
for chunk in *tree
lua = nomsu\compile(chunk)\as_statements("return ")
lua\declare_locals!
lua\prepend "-- File: #{source.filename\gsub("\n.*", "...")}\n"
if args.compile
output\write(tostring(lua), "\n")
if args.verbose
print(tostring(lua))
nomsu\run_lua(lua)
else
lua = nomsu\compile(tree)\as_statements("return ")
lua\declare_locals!
lua\prepend "-- File: #{source.filename\gsub("\n.*", "...")}\n"
if args.compile
output\write(tostring(lua), "\n")
if args.verbose
print(tostring(lua))
nomsu\run_lua(lua)
if args.compile
print ("Compiled %-25s -> %s")\format(filename, filename\gsub("%.nom$", ".lua"))
output\close!
2018-06-19 01:27:32 -07:00
if #parse_errs > 0
io.stderr\write table.concat(parse_errs, "\n\n")
io.stderr\flush!
os.exit(EXIT_FAILURE)
2018-06-19 01:27:32 -07:00
elseif args.syntax
os.exit(EXIT_SUCCESS)
2018-06-19 01:27:32 -07:00
if args.interactive
-- REPL
2018-06-23 18:26:18 -07:00
nomsu\run [[
use "core"
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
io.write(colored.bright colored.yellow ">> ")
buff = {}
while true
2018-06-23 00:57:31 -07:00
io.write(colors.bright)
2018-06-19 01:27:32 -07:00
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
if #buff > 0
io.write("\027[1A\027[2K")
break -- Run buffer
line = line\gsub("\t", " ")
table.insert buff, line
io.write(colored.dim colored.yellow ".. ")
if #buff == 0
break -- Exit
buff = table.concat(buff)
2018-06-19 15:24:24 -07:00
pseudo_filename = "user input #"..repl_line
files.spoof(pseudo_filename, buff)
2018-06-19 01:27:32 -07:00
err_hand = (error_message)->
2018-06-19 15:24:24 -07:00
Errhand.print_error error_message
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
print "= "..repr(ret)
elseif not ok
2018-06-19 15:24:24 -07:00
Errhand.print_error ret
2018-06-19 01:27:32 -07:00
2018-06-19 02:00:52 -07:00
has_ldt, ldt = pcall(require,'ldt')
if has_ldt
ldt.guard(run)
else
2018-06-19 15:24:24 -07:00
Errhand.run_safely(run)