moonscript/moonc

296 lines
5.4 KiB
Plaintext
Raw Normal View History

2011-08-13 04:07:44 +00:00
#!/usr/bin/env lua
module("moonscript", package.seeall)
require "moonscript.parse"
2011-07-04 16:20:46 +00:00
require "moonscript.compile"
require "moonscript.util"
require "alt_getopt"
require "lfs"
2011-08-13 04:07:44 +00:00
local opts, ind = alt_getopt.get_opts(arg, "vhwt:pTb", {
print = "p", tree = "T", version = "v", help = "h"
})
2011-08-13 04:07:44 +00:00
local help = [[Usage: %s [options] files...
-h Print this message
-w Watch file/directory
-t path Specify where to place compiled files
2011-08-13 04:07:44 +00:00
-p Write output to standard out
-T Write parse tree instead of code (to stdout)
-b Dump parse and compile time (doesn't write output)
2011-07-19 06:53:43 +00:00
-v Print version
]]
2011-07-19 06:53:43 +00:00
if opts.v then
local v = require "moonscript.version"
v.print_version()
os.exit()
end
function print_help(err)
if err then print("Error: "..err) end
print(help:format(arg[0]))
os.exit()
end
function mkdir(path)
local chunks = util.split(path, "/")
local accum
for _, dir in ipairs(chunks) do
accum = accum and accum.."/"..dir or dir
lfs.mkdir(accum)
end
return lfs.attributes(path, "mode")
end
function normalize(path)
return path:match("(.-)/*$").."/"
end
function get_dir(fname)
return fname:match("^(.-)[^/]*$")
end
-- convert .moon to .lua
function convert_path(path)
return (path:gsub("%.moon$", ".lua"))
end
2011-08-13 04:07:44 +00:00
function msg(...)
if not opts.p then
print(...)
end
2011-08-13 04:07:44 +00:00
end
local gettime = nil
if opts.b then
pcall(function()
require "socket"
gettime = socket.gettime
end)
function format_time(time)
return ("%.3fms"):format(time*1000)
end
if not gettime then
print_help"LuaSocket needed for benchmark"
end
else
gettime = function() return 0 end
end
function write_file(fname, code)
if opts.p then
if code ~= "" then print(code) end
else
mkdir(get_dir(fname))
local out_f = io.open(fname, "w")
if not out_f then
return nil, "Failed to write output: "..fname
end
out_f:write(code.."\n")
out_f:close()
end
return true
end
2011-08-13 04:07:44 +00:00
function compile_file(text, fname)
local parse_time = gettime()
local tree, err = parse.string(text)
2011-08-13 04:07:44 +00:00
parse_time = gettime() - parse_time
if not tree then
return nil, err
end
2011-08-13 04:07:44 +00:00
if opts.T then
opts.p = true
dump.tree(tree)
return ""
else
local compile_time = gettime()
local code, err, pos = compile.tree(tree)
compile_time = gettime() - compile_time
if not code then
return nil, compile.format_error(err, pos, text)
end
if opts.b then
opts.p = true
return table.concat({
fname,
"Parse time \t" .. format_time(parse_time),
"Compile time\t" .. format_time(compile_time),
""
}, "\n")
end
return code
end
2011-08-13 04:07:44 +00:00
end
2011-08-13 04:07:44 +00:00
function compile_and_write(from, to)
local f = io.open(from)
if not f then
return nil, "Can't find file"
end
2011-08-13 04:07:44 +00:00
local text = f:read("*a")
2011-08-13 04:07:44 +00:00
local code, err = compile_file(text, from)
if not code then
return nil, err
end
return write_file(to, code)
end
function scan_directory(root, collected)
root = normalize(root)
collected = collected or {}
for fname in lfs.dir(root) do
if not fname:match("^%.") then
local full_path = root..fname
if lfs.attributes(full_path, "mode") == "directory" then
scan_directory(full_path, collected)
end
if fname:match(".moon") then
table.insert(collected, full_path)
end
end
end
return collected
end
function append(a, b)
for _, v in ipairs(b) do
table.insert(a, v)
end
end
function remove_dups(tbl)
local hash = {}
local final = {}
for _, v in ipairs(tbl) do
if not hash[v] then
table.insert(final, v)
hash[v] = true
end
end
return final
end
function get_files(fname, files)
files = files or {}
if lfs.attributes(fname, "mode") == "directory" then
append(files, scan_directory(fname))
else
table.insert(files, "./"..fname)
end
return files
end
if opts.h then print_help() end
local inputs = {}
for i = ind, #arg do
table.insert(inputs, arg[i])
end
if #inputs == 0 then
print_help("No files specified")
end
local target_dir = "."
if opts.t then
if mkdir(opts.t) ~= "directory" then
print_help("Invalid target dir")
end
target_dir = opts.t
end
target_dir = target_dir.."/"
local files = {}
for _, input in ipairs(inputs) do
get_files(input, files)
end
files = remove_dups(files)
if opts.w then
local inotify
if not pcall(function()
inotify = require "inotify"
end) then
print_help("inotify not installed")
end
local dirs = {}
for _, fname in ipairs(files) do
table.insert(dirs, get_dir(fname))
end
dirs = remove_dups(dirs)
print(("Starting watch loop, Ctrl-C to exit [%d dirs]"):format(#dirs))
local wd_table = {}
local handle = inotify:init()
for _, dir in ipairs(dirs) do
local wd = handle:addwatch(dir, inotify.IN_MODIFY)
wd_table[wd] = dir
end
while true do
local success, events = pcall(handle.read, handle)
if not success then
print"\nQuitting..."
break
end
if events then
for _, ev in ipairs(events) do
local fname = wd_table[ev.wd]..ev.name
2011-05-30 08:23:35 +00:00
if fname:match("%.moon$") then
local target = target_dir..convert_path(fname)
2011-08-13 04:07:44 +00:00
local success, err = compile_and_write(fname, target)
2011-05-30 08:23:35 +00:00
if not success then
print()
print("Error: "..fname)
print(err)
print()
else
2011-08-13 04:07:44 +00:00
msg("Built", fname)
2011-05-30 08:23:35 +00:00
end
end
end
else break end
end
else
for _, fname in ipairs(files) do
2011-08-13 04:07:44 +00:00
local success, err = compile_and_write(fname, target_dir..convert_path(fname))
if not success then
print(fname, err)
break
2011-05-30 08:23:35 +00:00
else
2011-08-13 04:07:44 +00:00
msg("Built", fname)
end
end
end