nomsu/nomsu.moon

162 lines
6.8 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"
--NomsuCompiler = require "nomsu_compiler"
2018-06-19 01:27:32 -07:00
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
--{:Importer, :import_to_1_from, :_1_forked} = require 'importer'
{:List, :Dict, :Text} = require 'containers'
--SyntaxTree = require "syntax_tree"
nomsu_environment = require('nomsu_environment')
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 nomsu_environment
2018-06-19 01:27:32 -07:00
file_queue = List{}
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)
add_file: (f)-> file_queue\add(f)
add_exec_string: (pos, s)->
name = "command line arg @#{pos}.nom"
Files.spoof(name, s)
file_queue\add 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)
nomsu_environment.command_line_args = List(args.nomsu_args)
2018-11-11 18:03:18 -08:00
nomsu_environment.OPTIMIZATION = tonumber(args.optimization or 1)
2018-06-19 01:27:32 -07:00
2018-06-23 00:57:31 -07:00
if args.version
2018-11-11 18:03:18 -08:00
nomsu_environment.run_file_1_in 'core', nomsu_environment, nomsu_environment.OPTIMIZATION
nomsu_environment.run_1_in([[say (Nomsu version)]], nomsu_environment)
2018-06-23 00:57:31 -07:00
os.exit(EXIT_SUCCESS)
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
unless args.no_core
2018-11-11 18:03:18 -08:00
nomsu_environment.run_file_1_in 'core', nomsu_environment, nomsu_environment.OPTIMIZATION
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
code = Files.read(filename)
source = Source(filename, 1, #code)
nomsu_environment._1_parsed(NomsuCode\from(source, code))
2018-07-09 16:58:46 -07:00
print("Parse succeeded: #{filename}")
elseif 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")
code = Files.read(filename)
source = Source(filename, 1, #code)
code = NomsuCode\from(source, code)
tree = nomsu_environment._1_parsed(code)
tree = {tree} unless tree.type == 'FileChunks'
for chunk in *tree
lua = nomsu_environment.compile(chunk)
lua\declare_locals!
nomsu_environment.run_1_in(chunk, nomsu_environment)
2018-07-09 16:58:46 -07:00
output\write(tostring(lua), "\n")
if args.verbose then print(tostring(lua))
print ("Compiled %-25s -> %s")\format(filename, filename\gsub("%.nom$", ".lua"))
output\close!
elseif args.verbose
code = Files.read(filename)
source = Source(filename, 1, #code)
code = NomsuCode\from(source, code)
tree = nomsu_environment._1_parsed(code)
tree = {tree} unless tree.type == 'FileChunks'
for chunk in *tree
lua = nomsu_environment.compile(chunk)
lua\declare_locals!
nomsu_environment.run_1_in(chunk, nomsu_environment)
print(tostring(lua))
else
2018-07-09 16:58:46 -07:00
-- Just run the file
nomsu_environment.run_file_1_in(filename, nomsu_environment, 0)
2018-06-19 01:27:32 -07:00
unless args.primary_file or args.exec_strings
2018-11-11 18:03:18 -08:00
nomsu_environment.run_file_1_in("tools/repl.nom", nomsu_environment, nomsu_environment.OPTIMIZATION)
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)