2011-08-13 04:07:44 +00:00
|
|
|
#!/usr/bin/env lua
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2012-11-29 04:53:47 +00:00
|
|
|
local parse = require "moonscript.parse"
|
2012-11-28 08:42:43 +00:00
|
|
|
local compile = require "moonscript.compile"
|
2012-11-28 08:24:58 +00:00
|
|
|
local util = require "moonscript.util"
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2013-01-11 01:53:47 +00:00
|
|
|
local dump_tree = require"moonscript.dump".tree
|
|
|
|
|
2013-02-28 03:17:16 +00:00
|
|
|
local alt_getopt = require "alt_getopt"
|
|
|
|
local lfs = require "lfs"
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2014-06-17 17:45:42 +00:00
|
|
|
local dirsep = package.config:sub(1,1)
|
|
|
|
|
2014-05-28 05:33:24 +00:00
|
|
|
local opts, ind = alt_getopt.get_opts(arg, "lvhwt:o:pTXb", {
|
2013-12-26 09:29:00 +00:00
|
|
|
print = "p", tree = "T", version = "v", help = "h", lint = "l"
|
2011-08-13 04:07:44 +00:00
|
|
|
})
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2012-01-22 19:53:13 +00:00
|
|
|
local read_stdin = arg[1] == "--"
|
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
local polling_rate = 1.0
|
|
|
|
|
2011-08-13 04:07:44 +00:00
|
|
|
local help = [[Usage: %s [options] files...
|
2011-05-29 16:33:25 +00:00
|
|
|
|
|
|
|
-h Print this message
|
|
|
|
-w Watch file/directory
|
|
|
|
-t path Specify where to place compiled files
|
2014-05-28 05:33:24 +00:00
|
|
|
-o file Write output to file
|
2011-08-13 04:07:44 +00:00
|
|
|
-p Write output to standard out
|
|
|
|
-T Write parse tree instead of code (to stdout)
|
2012-10-30 07:31:56 +00:00
|
|
|
-X Write line rewrite map instead of code (to stdout)
|
2013-12-26 09:29:00 +00:00
|
|
|
-l Perform lint on the file instead of compiling
|
2011-08-13 04:07:44 +00:00
|
|
|
-b Dump parse and compile time (doesn't write output)
|
2011-07-19 06:53:43 +00:00
|
|
|
-v Print version
|
2012-01-22 19:53:13 +00:00
|
|
|
|
|
|
|
-- Read from standard in, print to standard out
|
|
|
|
(Must be first and only argument)
|
2011-05-29 16:33:25 +00:00
|
|
|
]]
|
|
|
|
|
2011-07-19 06:53:43 +00:00
|
|
|
if opts.v then
|
|
|
|
local v = require "moonscript.version"
|
|
|
|
v.print_version()
|
|
|
|
os.exit()
|
|
|
|
end
|
|
|
|
|
2011-05-29 16:33:25 +00:00
|
|
|
function print_help(err)
|
2013-12-27 06:47:11 +00:00
|
|
|
local help_msg = help:format(arg[0])
|
|
|
|
|
2013-12-27 06:44:54 +00:00
|
|
|
if err then
|
|
|
|
io.stderr:write("Error: ".. err .. "\n")
|
2013-12-27 06:47:11 +00:00
|
|
|
io.stderr:write(help_msg .. "\n")
|
|
|
|
os.exit(1)
|
|
|
|
else
|
|
|
|
print(help_msg)
|
|
|
|
os.exit(0)
|
2013-12-27 06:44:54 +00:00
|
|
|
end
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
|
|
|
|
2013-12-27 06:44:54 +00:00
|
|
|
function log_msg(...)
|
2011-08-13 04:07:44 +00:00
|
|
|
if not opts.p then
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write(table.concat({...}, " ") .. "\n")
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
2011-08-13 04:07:44 +00:00
|
|
|
end
|
|
|
|
|
2014-06-18 16:23:41 +00:00
|
|
|
local moonc = require("moonscript.cmd.moonc")
|
|
|
|
local mkdir = moonc.mkdir
|
|
|
|
local normalize_dir = moonc.normalize_dir
|
|
|
|
local parse_dir = moonc.parse_dir
|
|
|
|
local parse_file = moonc.parse_file
|
|
|
|
local convert_path = moonc.convert_path
|
|
|
|
local compile_file_text = moonc.compile_file_text
|
2011-08-13 04:07:44 +00:00
|
|
|
|
|
|
|
function write_file(fname, code)
|
|
|
|
if opts.p then
|
|
|
|
if code ~= "" then print(code) end
|
|
|
|
else
|
2014-06-18 03:27:10 +00:00
|
|
|
mkdir(parse_dir(fname))
|
2011-08-13 04:07:44 +00:00
|
|
|
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
|
2012-10-30 07:31:56 +00:00
|
|
|
|
2014-06-18 16:23:41 +00:00
|
|
|
return "built"
|
2011-08-13 04:07:44 +00:00
|
|
|
end
|
2011-05-29 16:33:25 +00:00
|
|
|
|
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"
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
2011-08-13 04:07:44 +00:00
|
|
|
local text = f:read("*a")
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2014-06-18 16:23:41 +00:00
|
|
|
local code, err = compile_file_text(text, from, {
|
|
|
|
benchmark = opts.b,
|
|
|
|
show_posmap = opts.X,
|
|
|
|
show_parse_tree = opts.T,
|
|
|
|
})
|
|
|
|
|
|
|
|
if not code and err then
|
2011-08-13 04:07:44 +00:00
|
|
|
return nil, err
|
|
|
|
end
|
|
|
|
|
2014-06-18 16:23:41 +00:00
|
|
|
if not code then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
2011-08-13 04:07:44 +00:00
|
|
|
return write_file(to, code)
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function scan_directory(root, collected)
|
2014-06-18 16:23:41 +00:00
|
|
|
root = normalize_dir(root)
|
2011-05-29 16:33:25 +00:00
|
|
|
collected = collected or {}
|
|
|
|
|
|
|
|
for fname in lfs.dir(root) do
|
2014-06-17 07:03:13 +00:00
|
|
|
if not fname:match("^%.") then
|
2011-05-29 16:33:25 +00:00
|
|
|
local full_path = root..fname
|
|
|
|
|
|
|
|
if lfs.attributes(full_path, "mode") == "directory" then
|
|
|
|
scan_directory(full_path, collected)
|
|
|
|
end
|
|
|
|
|
2013-01-14 07:38:38 +00:00
|
|
|
if fname:match("%.moon$") then
|
2011-05-29 16:33:25 +00:00
|
|
|
table.insert(collected, full_path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return collected
|
|
|
|
end
|
|
|
|
|
2014-06-17 17:45:42 +00:00
|
|
|
function remove_dups(tbl, key_fn)
|
2011-05-29 16:33:25 +00:00
|
|
|
local hash = {}
|
|
|
|
local final = {}
|
|
|
|
|
|
|
|
for _, v in ipairs(tbl) do
|
2014-06-17 17:45:42 +00:00
|
|
|
local dup_key = key_fn and key_fn(v) or v
|
|
|
|
if not hash[dup_key] then
|
2011-05-29 16:33:25 +00:00
|
|
|
table.insert(final, v)
|
2014-06-17 17:45:42 +00:00
|
|
|
hash[dup_key] = true
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return final
|
|
|
|
end
|
|
|
|
|
2014-06-18 03:27:10 +00:00
|
|
|
local function is_abs_path(path)
|
|
|
|
local first = path:sub(1, 1)
|
|
|
|
if dirsep == "\\" then
|
|
|
|
return first == "/" or first == "\\" or path:sub(2,1) == ":"
|
|
|
|
else
|
|
|
|
return first == dirsep
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-06-17 17:45:42 +00:00
|
|
|
-- creates tuples of input and target
|
2011-05-29 16:33:25 +00:00
|
|
|
function get_files(fname, files)
|
|
|
|
files = files or {}
|
|
|
|
|
|
|
|
if lfs.attributes(fname, "mode") == "directory" then
|
2014-06-17 17:45:42 +00:00
|
|
|
local head = fname:match("^(.-)[^" .. dirsep .. "]*" .. dirsep .."?$")
|
|
|
|
for _, sub_fname in ipairs(scan_directory(fname)) do
|
|
|
|
local target_fname = convert_path(sub_fname)
|
|
|
|
if opts.t then
|
|
|
|
if head then
|
|
|
|
local start, stop = target_fname:find(head, 1, true)
|
|
|
|
if start == 1 then
|
|
|
|
target_fname = target_fname:sub(stop + 1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-06-18 16:23:41 +00:00
|
|
|
target_fname = normalize_dir(opts.t) .. target_fname
|
2014-06-17 17:45:42 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
table.insert(files, {sub_fname, target_fname})
|
|
|
|
end
|
2011-05-29 16:33:25 +00:00
|
|
|
else
|
2014-06-17 17:45:42 +00:00
|
|
|
local target_fname = convert_path(fname)
|
|
|
|
if opts.t then
|
2014-06-18 16:23:41 +00:00
|
|
|
local prefix = normalize_dir(opts.t)
|
2014-06-18 03:27:10 +00:00
|
|
|
|
|
|
|
if is_abs_path(target_fname) then
|
|
|
|
target_fname = parse_file(target_fname)
|
|
|
|
end
|
|
|
|
target_fname = prefix .. target_fname
|
2014-06-17 17:45:42 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
table.insert(files, {fname, target_fname})
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return files
|
|
|
|
end
|
|
|
|
|
|
|
|
if opts.h then print_help() end
|
|
|
|
|
2012-01-22 19:53:13 +00:00
|
|
|
if read_stdin then
|
|
|
|
local text = io.stdin:read("*a")
|
|
|
|
local tree, err = parse.string(text)
|
|
|
|
if not tree then error(err) end
|
|
|
|
local code, err, pos = compile.tree(tree)
|
|
|
|
|
|
|
|
if not code then
|
|
|
|
error(compile.format_error(err, pos, text))
|
|
|
|
end
|
|
|
|
|
|
|
|
print(code)
|
|
|
|
os.exit()
|
|
|
|
end
|
|
|
|
|
2011-05-29 16:33:25 +00:00
|
|
|
local inputs = {}
|
|
|
|
for i = ind, #arg do
|
|
|
|
table.insert(inputs, arg[i])
|
|
|
|
end
|
|
|
|
|
|
|
|
if #inputs == 0 then
|
|
|
|
print_help("No files specified")
|
|
|
|
end
|
|
|
|
|
|
|
|
local files = {}
|
|
|
|
for _, input in ipairs(inputs) do
|
|
|
|
get_files(input, files)
|
|
|
|
end
|
|
|
|
|
2014-06-18 03:27:10 +00:00
|
|
|
files = remove_dups(files, function(f)
|
|
|
|
return f[2]
|
|
|
|
end)
|
2014-06-17 17:45:42 +00:00
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
function get_sleep_func()
|
|
|
|
local sleep
|
|
|
|
if not pcall(function()
|
|
|
|
require "socket"
|
|
|
|
sleep = socket.sleep
|
|
|
|
end) then
|
2013-01-03 01:32:00 +00:00
|
|
|
-- This is set by moonc.c in windows binaries
|
|
|
|
sleep = require("moonscript")._sleep
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
2011-12-01 03:25:41 +00:00
|
|
|
if not sleep then
|
|
|
|
error("Missing sleep function; install LuaSocket")
|
|
|
|
end
|
2011-10-09 08:59:50 +00:00
|
|
|
return sleep
|
|
|
|
end
|
|
|
|
|
2011-10-09 21:52:50 +00:00
|
|
|
function plural(count, word)
|
|
|
|
if count ~= 1 then
|
|
|
|
word = word .. "s"
|
|
|
|
end
|
|
|
|
return table.concat({count, word}, " ")
|
|
|
|
end
|
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
-- returns an iterator that returns files that have been updated
|
|
|
|
function create_watcher(files)
|
2011-12-01 03:25:41 +00:00
|
|
|
local msg = "Starting watch loop (Ctrl-C to exit)"
|
2011-10-09 08:59:50 +00:00
|
|
|
|
2011-05-29 16:33:25 +00:00
|
|
|
local inotify
|
2011-10-09 09:03:09 +00:00
|
|
|
pcall(function()
|
2011-05-29 16:33:25 +00:00
|
|
|
inotify = require "inotify"
|
2011-10-09 09:03:09 +00:00
|
|
|
end)
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2011-10-09 09:03:09 +00:00
|
|
|
if inotify then
|
2011-10-09 08:59:50 +00:00
|
|
|
local dirs = {}
|
2014-06-18 03:27:10 +00:00
|
|
|
|
|
|
|
for _, tuple in ipairs(files) do
|
|
|
|
local dir = parse_dir(tuple[1])
|
|
|
|
if dir == "" then
|
|
|
|
dir = "./"
|
|
|
|
end
|
|
|
|
table.insert(dirs, dir)
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
2014-06-18 03:27:10 +00:00
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
dirs = remove_dups(dirs)
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
return coroutine.wrap(function()
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write(("%s with inotify [%s]"):format(msg, plural(#dirs, "dir")) .. "\n")
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
local wd_table = {}
|
2012-03-01 03:32:10 +00:00
|
|
|
local handle = inotify.init()
|
2011-10-09 08:59:50 +00:00
|
|
|
for _, dir in ipairs(dirs) do
|
2014-01-04 23:41:20 +00:00
|
|
|
local wd = handle:addwatch(dir, inotify.IN_CLOSE_WRITE, inotify.IN_MOVED_TO)
|
2011-10-09 08:59:50 +00:00
|
|
|
wd_table[wd] = dir
|
|
|
|
end
|
2011-05-29 16:33:25 +00:00
|
|
|
|
2011-10-09 08:59:50 +00:00
|
|
|
while true do
|
2011-10-09 21:52:50 +00:00
|
|
|
local events = handle:read()
|
2014-06-18 03:27:10 +00:00
|
|
|
if not events then
|
|
|
|
break
|
|
|
|
end
|
|
|
|
|
|
|
|
for _, ev in ipairs(events) do
|
|
|
|
local fname = ev.name
|
|
|
|
if fname:match("%.moon$") then
|
|
|
|
local dir = wd_table[ev.wd]
|
|
|
|
if dir ~= "./" then
|
|
|
|
fname = dir .. fname
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
2014-06-18 03:27:10 +00:00
|
|
|
-- TODO: check to make sure the file was in the original set
|
|
|
|
coroutine.yield(fname)
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
else
|
|
|
|
-- poll the filesystem instead
|
|
|
|
local sleep = get_sleep_func()
|
|
|
|
return coroutine.wrap(function()
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write(("%s with polling [%s]"):format(msg, plural(#files, "file")) .. "\n")
|
2011-10-09 08:59:50 +00:00
|
|
|
|
|
|
|
local mod_time = {}
|
|
|
|
while true do
|
2014-06-18 03:27:10 +00:00
|
|
|
for _, tuple in ipairs(files) do
|
|
|
|
local file = tuple[1]
|
2011-10-09 08:59:50 +00:00
|
|
|
local time = lfs.attributes(file, "modification")
|
|
|
|
if not mod_time[file] then
|
|
|
|
mod_time[file] = time
|
2011-05-30 08:23:35 +00:00
|
|
|
else
|
2011-10-09 08:59:50 +00:00
|
|
|
if time ~= mod_time[file] then
|
|
|
|
if time > mod_time[file] then
|
|
|
|
coroutine.yield(file)
|
|
|
|
mod_time[file] = time
|
|
|
|
end
|
|
|
|
end
|
2011-05-30 08:23:35 +00:00
|
|
|
end
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
2011-10-09 08:59:50 +00:00
|
|
|
sleep(polling_rate)
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
2011-10-09 08:59:50 +00:00
|
|
|
end)
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
if opts.w then
|
|
|
|
local watcher = create_watcher(files)
|
|
|
|
-- catches interrupt error for ctl-c
|
|
|
|
local protected = function()
|
2014-06-18 03:27:10 +00:00
|
|
|
local status, file = true, watcher()
|
2011-12-01 03:25:41 +00:00
|
|
|
if status then
|
|
|
|
return file
|
|
|
|
elseif file ~= "interrupted!" then
|
|
|
|
error(file)
|
|
|
|
end
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
for fname in protected do
|
2014-06-18 03:27:10 +00:00
|
|
|
local target = convert_path(fname)
|
2011-10-09 08:59:50 +00:00
|
|
|
local success, err = compile_and_write(fname, target)
|
|
|
|
if not success then
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write(table.concat({
|
|
|
|
"",
|
|
|
|
"Error: " .. fname,
|
|
|
|
err,
|
|
|
|
"\n",
|
|
|
|
}, "\n"))
|
2014-06-18 16:23:41 +00:00
|
|
|
elseif success == "build" then
|
|
|
|
log_msg("Built", fname, "->", target)
|
2011-10-09 08:59:50 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write("\nQuitting...\n")
|
2013-12-26 09:29:00 +00:00
|
|
|
elseif opts.l then
|
2014-06-18 03:27:10 +00:00
|
|
|
for _, tuple in pairs(files) do
|
|
|
|
local fname = tuple[1]
|
2013-12-26 09:29:00 +00:00
|
|
|
lint = require "moonscript.cmd.lint"
|
2013-12-27 23:32:06 +00:00
|
|
|
local res, err = lint.lint_file(fname)
|
2013-12-26 09:29:00 +00:00
|
|
|
if res then
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write(res .. "\n\n")
|
2013-12-27 23:32:06 +00:00
|
|
|
elseif err then
|
|
|
|
io.stderr:write(fname .. "\n" .. err.. "\n\n")
|
2013-12-26 09:29:00 +00:00
|
|
|
end
|
|
|
|
end
|
2011-05-29 16:33:25 +00:00
|
|
|
else
|
2014-06-18 03:27:10 +00:00
|
|
|
for _, tuple in ipairs(files) do
|
|
|
|
local fname, target = unpack(tuple)
|
2014-05-28 05:33:24 +00:00
|
|
|
local success, err = compile_and_write(fname, target)
|
2011-05-29 16:33:25 +00:00
|
|
|
if not success then
|
2013-12-27 06:44:54 +00:00
|
|
|
io.stderr:write(fname .. "\t" .. err .. "\n")
|
2011-09-19 06:23:23 +00:00
|
|
|
os.exit(1)
|
2014-06-18 16:23:41 +00:00
|
|
|
elseif success == "build" then
|
|
|
|
log_msg("Built", fname)
|
2011-05-29 16:33:25 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|