Cleanup of some metaprogramming stuff, as well as adding support for

"package.nomsupath" to search for files in different locations, and
prioritizing use of "luafilesystem" over system calls.
This commit is contained in:
Bruce Hill 2018-06-21 19:12:59 -07:00
parent 7761f715f7
commit 86a3219e7f
12 changed files with 244 additions and 46 deletions

View File

@ -383,3 +383,18 @@ compile [do %action then always %final_action] to
# Inline thunk:
compile [result of %body] to
Lua value "(\(compile as: [] -> %body))()"
# Recurion control flow
using
compile [%var's stack] to: Lua value "stack\(%var as lua id)"
..compile
parse [for %var in recursive %structure %body] as
with local {(%var's stack): [%structure], action: recurse % on %}
action [recurse %v on %x]
add %x to (%v's stack)
repeat while: (length of (%var's stack)) > 0
%var <- (remove 1 from (%var's stack))
%body
=== next %var ==
=== stop %var ===

View File

@ -52,6 +52,20 @@ lua> ".."
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
compile [using %defs compile %body] to
lua> ".."
local lua = LuaCode(tree.source)
lua:append(
"local old_nomsu = nomsu\n",
"local nomsu = table.fork(old_nomsu, {COMPILE_ACTIONS=table.fork(old_nomsu.COMPILE_ACTIONS)})")
lua:append(nomsu:compile(\%defs))
lua:append("\n")
lua:append("local ret = nomsu:compile(tree)\n")
lua:append("return ret")
nomsu = table.fork(nomsu, {tree=\%body})
local output = nomsu:run_lua(lua)
return output
compile [local action %actions %body] to
lua> ".."
local fn_name = "A"..string.as_lua_id(\%actions[1].stub)
@ -84,6 +98,9 @@ compile [action %actions %body] to
lua:remove_free_vars(table.map(\%actions, function(a) return "A"..string.as_lua_id(a.stub) end))
return lua
compile [action %action] to
Lua value "A\(%action.stub as lua id)"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
compile [parse %actions as %body] to
@ -160,6 +177,9 @@ compile [declare locals in %code] to
compile [declare locals %locals in %code] to
Lua value "\(%code as lua expr):declare_locals(\(%locals as lua expr))"
compile [add free vars %vars to %code] to
Lua "\(%code as lua expr):add_free_vars(\(%vars as lua expr));"
compile [remove free vars %vars from %code] to
Lua "\(%code as lua expr):remove_free_vars(\(%vars as lua expr));"

View File

@ -130,16 +130,6 @@ compile [with %assignments %body] to
\%lua
end -- 'with' block
compile [local %var_or_vars] to
%lua <- (Lua "")
lua> ".."
if \%var_or_vars.type == "List" then
\%lua:add_free_vars(table.map(\%var_or_vars, function(v) return tostring(nomsu:compile(v)) end))
else
\%lua:add_free_vars({tostring(nomsu:compile(\%var_or_vars))})
end
return %lua
# Math Operators
compile [%x wrapped around %y, %x mod %y] to: Lua value "(\(%x as lua expr) % \(%y as lua expr))"

View File

@ -9,18 +9,18 @@ compile [with local %locals %body, with local %locals do %body] to
* "Dict"
%body_lua <-
Lua ".."
\(=lua "A_assign_1(\%locals, \%locals)")
\(compile as: <- %locals)
\%body_lua
declare locals
(%.1 as lua id) for % in %locals
"\(%.1 as lua)" for % in %locals
.. in %body_lua
* "List"
declare locals
(% as lua id) for % in %locals
"\(% as lua)" for % in %locals
.. in %body_lua
* "Var"
* "Action"
declare locals [%locals as lua id] in %body_lua
declare locals ["\(%locals as lua)"] in %body_lua
* else
barf "Unexpected local: \(%locals as nomsu)"
return

View File

@ -197,7 +197,8 @@ end
local error_handler
error_handler = function(error_message)
print_error(error_message)
return os.exit(false, true)
local EXIT_FAILURE = 1
return os.exit(EXIT_FAILURE)
end
local run_safely
run_safely = function(fn)

View File

@ -123,7 +123,8 @@ print_error = (error_message, stack_offset=3)->
error_handler = (error_message)->
print_error error_message
os.exit(false, true)
EXIT_FAILURE = 1
os.exit(EXIT_FAILURE)
run_safely = (fn)->
xpcall(fn, error_handler)

View File

@ -4,18 +4,18 @@
use "core"
compile [@, me] to: Lua value "self"
compile [set methods %methods] to
%lua <- (Lua "")
for %m in %methods
to %lua write "\nclass.\(%m as lua id) = \(%m as lua id)"
return %lua
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
parse [method %actions %body] as
with local %actions
action %actions %body
set methods %actions
compile [method %actions %body] to
%lua <- (compile as: action %actions %body)
add free vars ((% as lua id) for % in %actions) to %lua
declare locals in %lua
for % in %actions
to %lua write "\nclass.\(% as lua id) = \(% as lua id)"
return
Lua ".."
do -- Method: \(%actions.1.stub)
\%lua
end
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
parse [as %instance %body] as

114
nomsu.lua
View File

@ -1,3 +1,4 @@
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
local usage = [=[Nomsu Compiler
Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-v] [-c] [-f] [-s] [--help] [-p print_file] file1 file2... [-- nomsu args...]
@ -13,8 +14,14 @@ OPTIONS
-p <file> Print to the specified file instead of stdout.
<input> Input file can be "-" to use stdin.
]=]
local lpeg = require('lpeg')
local re = require('re')
local ok, _ = pcall(function()
lpeg = require('lpeg')
re = require('re')
end)
if not ok then
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)
end
local Errhand = require("error_handling")
local NomsuCompiler = require("nomsu_compiler")
local NomsuCode, LuaCode, Source
@ -46,10 +53,101 @@ local args = table.concat(arg, ";") .. ";"
args = parser:match(args)
if not args or args.help then
print(usage)
os.exit()
os.exit(EXIT_FAILURE)
end
local nomsu = NomsuCompiler
nomsu.arg = args.nomsu_args
FILE_CACHE = setmetatable({ }, {
__index = function(self, filename)
local file = io.open(filename)
if not (file) then
return nil
end
local contents = file:read("*a")
file:close()
self[filename] = contents
return contents
end
})
local match, sub, rep, gsub, format, byte, find
do
local _obj_0 = string
match, sub, rep, gsub, format, byte, match, find = _obj_0.match, _obj_0.sub, _obj_0.rep, _obj_0.gsub, _obj_0.format, _obj_0.byte, _obj_0.match, _obj_0.find
end
local iterate_single
iterate_single = function(item, prev)
if item == prev then
return nil
else
return item
end
end
local lfs
ok, lfs = pcall(require, "lfs")
if ok then
all_files = function(path)
local browse
browse = function(filename)
local file_type = lfs.attributes(filename, 'mode')
if file_type == 'file' then
if match(filename, "%.nom$") or match(filename, "%.lua$") then
coroutine.yield(filename)
return true
end
elseif file_type == 'directory' then
for subfile in lfs.dir(filename) do
if not (subfile == "." or subfile == "..") then
browse(filename .. "/" .. subfile)
end
end
return true
elseif file_type == 'char device' then
coroutine.yield(filename)
return true
end
return false
end
return coroutine.wrap(function()
if not browse(path) and package.nomsupath then
browse(package.nomsupath .. "/" .. path)
end
return nil
end)
end
else
local ret = os.execute('find . -maxdepth 0')
if not (ret == true or ret == 0) then
error("Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: http://keplerproject.github.io/luafilesystem/ or `luarocks install luafilesystem`)", 0)
end
all_files = function(path)
if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") then
return iterate_single, path
end
path = gsub(path, "\\", "\\\\")
path = gsub(path, "`", "")
path = gsub(path, '"', '\\"')
path = gsub(path, "$", "")
return coroutine.wrap(function()
local f = io.popen('find -L "' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"')
local found = false
for line in f:lines() do
found = true
coroutine.yield(line)
end
if not found and package.nomsupath then
f:close()
f = io.popen('find -L "' .. package.nomsupath .. '/' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"')
for line in f:lines() do
coroutine.yield(line)
end
end
local success = f:close()
if not (success) then
return error("Invalid file path: " .. tostring(path))
end
end)
end
end
local run
run = function()
for i, input in ipairs(args.inputs) do
@ -120,7 +218,8 @@ run = function()
local filename = input_files[_index_0]
if args.syntax then
local file_contents = io.open(filename):read('*a')
local ok, err = pcall(nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents))
local err
ok, err = pcall(nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents))
if not ok then
table.insert(parse_errs, err)
elseif print_file then
@ -149,9 +248,9 @@ run = function()
if #parse_errs > 0 then
io.stderr:write(table.concat(parse_errs, "\n\n"))
io.stderr:flush()
os.exit(false, true)
os.exit(EXIT_FAILURE)
elseif args.syntax then
os.exit(true, true)
os.exit(EXIT_SUCCESS)
end
if args.interactive then
for repl_line = 1, math.huge do
@ -179,7 +278,8 @@ run = function()
err_hand = function(error_message)
return Errhand.print_error(error_message)
end
local ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source(pseudo_filename, 1, #buff))
local ret
ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source(pseudo_filename, 1, #buff))
if ok and ret ~= nil then
print("= " .. repr(ret))
elseif not ok then

View File

@ -1,5 +1,6 @@
#!/usr/bin/env moon
-- This file contains the command-line Nomsu runner.
EXIT_SUCCESS, EXIT_FAILURE = 0, 1
usage = [=[
Nomsu Compiler
@ -17,8 +18,13 @@ OPTIONS
<input> Input file can be "-" to use stdin.
]=]
lpeg = require 'lpeg'
re = require 're'
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)
Errhand = require "error_handling"
NomsuCompiler = require "nomsu_compiler"
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
@ -45,11 +51,78 @@ args = table.concat(arg, ";")..";"
args = parser\match(args)
if not args or args.help
print usage
os.exit!
os.exit(EXIT_FAILURE)
nomsu = NomsuCompiler
nomsu.arg = args.nomsu_args
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
}
export all_files
{:match, :sub, :rep, :gsub, :format, :byte, :match, :find} = string
iterate_single = (item, prev) -> if item == prev then nil else item
ok, lfs = pcall(require, "lfs")
if ok
all_files = (path)->
browse = (filename)->
file_type = lfs.attributes(filename, 'mode')
if file_type == 'file'
if match(filename, "%.nom$") or match(filename, "%.lua$")
coroutine.yield filename
return true
elseif file_type == 'directory'
for subfile in lfs.dir(filename)
unless subfile == "." or subfile == ".."
browse(filename.."/"..subfile)
return true
elseif file_type == 'char device'
coroutine.yield(filename)
return true
return false
return coroutine.wrap ->
if not browse(path) and package.nomsupath
browse(package.nomsupath.."/"..path)
return nil
else
ret = os.execute('find . -maxdepth 0')
unless ret == true or ret == 0
error "Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: http://keplerproject.github.io/luafilesystem/ or `luarocks install luafilesystem`)", 0
all_files = (path)->
-- Sanitize path
if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$")
return iterate_single, path
-- TODO: improve sanitization
path = gsub(path,"\\","\\\\")
path = gsub(path,"`","")
path = gsub(path,'"','\\"')
path = gsub(path,"$","")
return coroutine.wrap ->
f = io.popen('find -L "'..path..'" -not -path "*/\\.*" -type f -name "*.nom"')
found = false
for line in f\lines!
found = true
coroutine.yield(line)
if not found and package.nomsupath
f\close!
f = io.popen('find -L "'..package.nomsupath..'/'..path..'" -not -path "*/\\.*" -type f -name "*.nom"')
for line in f\lines!
coroutine.yield(line)
success = f\close!
unless success
error("Invalid file path: "..tostring(path))
run = ->
for i,input in ipairs args.inputs
if input == "-" then args.inputs[i] = STDIN
@ -126,9 +199,9 @@ run = ->
if #parse_errs > 0
io.stderr\write table.concat(parse_errs, "\n\n")
io.stderr\flush!
os.exit(false, true)
os.exit(EXIT_FAILURE)
elseif args.syntax
os.exit(true, true)
os.exit(EXIT_SUCCESS)
if args.interactive
-- REPL

View File

@ -127,7 +127,7 @@ do
ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]*
]])
local nomsu_peg = peg_tidier:match(io.open("nomsu.peg"):read('*a'))
local nomsu_peg = peg_tidier:match(io.open((package.nomsupath or '.') .. "/nomsu.peg"):read('*a'))
NOMSU_PATTERN = re.compile(nomsu_peg, NOMSU_DEFS)
end
local parse
@ -162,8 +162,7 @@ parse = function(nomsu_code, source)
end
errors = _accum_0
end
io.stderr:write("Errors occurred while parsing:\n\n", table.concat(errors, "\n\n"), '\n')
os.exit(1)
error("Errors occurred while parsing:\n\n" .. table.concat(errors, "\n\n"), 0)
end
return tree
end

View File

@ -98,7 +98,7 @@ NOMSU_PATTERN = do
ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]*
]]
nomsu_peg = peg_tidier\match(io.open("nomsu.peg")\read('*a'))
nomsu_peg = peg_tidier\match(io.open((package.nomsupath or '.').."/nomsu.peg")\read('*a'))
re.compile(nomsu_peg, NOMSU_DEFS)
parse = (nomsu_code, source=nil)->
@ -116,8 +116,7 @@ parse = (nomsu_code, source=nil)->
keys = utils.keys(userdata.errors)
table.sort(keys)
errors = [userdata.errors[k] for k in *keys]
io.stderr\write("Errors occurred while parsing:\n\n", table.concat(errors, "\n\n"), '\n')
os.exit(1)
error("Errors occurred while parsing:\n\n"..table.concat(errors, "\n\n"), 0)
return tree

View File

@ -7,7 +7,7 @@ with local %x
assume: %x = "outer"
action [foo] "outer foo"
with local (foo)
with local [action: foo]
action [foo] "inner foo"
assume: (foo) = "inner foo"
assume: (foo) = "outer foo"