nomsu/nomsu.moon

233 lines
8.7 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.
EXIT_SUCCESS, EXIT_FAILURE = 0, 1
if _VERSION == "Lua 5.1" and not jit
-- Cannot run on Lua5.1 because it doesn't have gotos.
print("Sorry, Nomsu does not run on Lua 5.1. Please use LuaJIT 2+ or Lua 5.2+")
os.exit(EXIT_FAILURE)
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],";")..";."
else
package.nomsupath = "."
2018-06-19 01:27:32 -07:00
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-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)
Files = require "files"
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-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
2018-07-09 16:58:46 -07:00
file_queue = {}
2018-09-08 01:05:59 -07:00
sep = "\3"
2018-06-19 01:27:32 -07:00
parser = re.compile([[
2018-09-12 15:31:59 -07:00
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-07-09 16:58:46 -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: (f)-> table.insert(file_queue, f)
add_exec_string: (pos, s)->
name = "command line arg @#{pos}.nom"
Files.spoof(name, s)
table.insert(file_queue, name)
2018-07-09 16:58:46 -07:00
})
arg_string = table.concat(arg, sep)..sep
2018-06-23 00:57:31 -07:00
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
nomsu = NomsuCompiler
2018-09-16 16:57:14 -07:00
nomsu.environment.arg = NomsuCompiler.environment._List(args.nomsu_args)
2018-06-19 01:27:32 -07:00
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 = ->
input_files = {}
2018-07-09 16:58:46 -07:00
for f in *file_queue
2018-07-24 17:36:45 -07:00
if f == 'stdin'
input_files[f] = true
continue
unless Files.exists(f)
error("Could not find: '#{f}'")
for _,filename in Files.walk(f)
2018-07-09 16:58:46 -07:00
input_files[filename] = true
2018-06-19 01:27:32 -07:00
2018-06-23 00:57:31 -07:00
nomsu.can_optimize = (f)->
return false if args.optimization == 0
2018-07-09 16:58:46 -07:00
return false if args.compile and input_files[f]
2018-06-23 00:57:31 -07:00
return true
unless args.no_core
nomsu\import_file('core')
2018-07-09 16:58:46 -07:00
get_file_and_source = (filename)->
local file, source
2018-07-24 17:36:45 -07:00
if filename == 'stdin' or filename\match("%.nom$")
file = Files.read(filename)
2018-07-09 16:58:46 -07:00
if not file
error("File does not exist: #{filename}", 0)
source = Source(filename, 1, #file)
else return nil
source = Source(filename,1,#file)
return file, source
run_file = (filename, lua_handler=nil)->
file, source = get_file_and_source(filename)
return unless file
2018-07-09 16:58:46 -07:00
tree = nomsu\parse(file, source)
if tree
if tree.type != "FileChunks"
2018-07-09 16:58:46 -07:00
tree = {tree}
-- 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 lua_handler and input_files[filename] then lua_handler(tostring(lua))
2018-07-09 16:58:46 -07:00
nomsu\run_lua(lua)
2018-06-19 01:27:32 -07:00
parse_errs = {}
2018-07-09 16:58:46 -07:00
for f in *file_queue
for _,filename in Files.walk(f)
continue unless filename == "stdin" or filename\match("%.nom$")
2018-07-09 16:58:46 -07:00
if args.check_syntax
-- Check syntax
file, source = get_file_and_source(filename)
continue unless file
2018-09-12 15:31:59 -07:00
tree = nomsu\parse(file, source)
2018-07-09 16:58:46 -07:00
print("Parse succeeded: #{filename}")
if args.compile
2018-07-09 16:58:46 -07:00
-- Compile .nom files into .lua
output = if filename == 'stdin' then io.output()
else io.open(filename\gsub("%.nom$", ".lua"), "w")
run_file filename, (lua)->
output\write(tostring(lua), "\n")
if args.verbose then print(tostring(lua))
print ("Compiled %-25s -> %s")\format(filename, filename\gsub("%.nom$", ".lua"))
output\close!
2018-06-19 01:27:32 -07:00
2018-07-09 16:58:46 -07:00
if not args.check_syntax and not args.compile
-- Just run the file
run_file filename, (args.verbose and print or nil)
2018-06-19 01:27:32 -07:00
unless args.primary_file or args.exec_strings
2018-07-09 16:58:46 -07:00
-- Run in interactive mode (REPL)
2018-06-23 18:26:18 -07:00
nomsu\run [[
2018-09-14 19:17:09 -07:00
#!/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
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, NomsuCode(Source(pseudo_filename,1,#buff), buff)
2018-06-19 01:27:32 -07:00
if ok and ret != nil
if type(ret) == 'number'
print "= #{ret}"
else
print "= #{ret\as_nomsu!}"
2018-06-19 01:27:32 -07:00
elseif not ok
2018-06-19 15:24:24 -07:00
Errhand.print_error ret
2018-06-19 01:27:32 -07:00
2018-07-17 16:15:44 -07:00
debugger = if args.debugger == "nil" then {}
else require(args.debugger or 'error_handling')
guard = if type(debugger) == 'function' then debugger
else debugger.guard or debugger.call or debugger.wrap or debugger.run or ((fn)->fn())
guard(run)