Improvements to nomsu codegen (comments are now retained) and some
improvements to handling of stdin and fixes for error reporting.
This commit is contained in:
parent
09d6bad6ac
commit
0923b0192c
11
code_obj.lua
11
code_obj.lua
@ -82,8 +82,14 @@ do
|
|||||||
local bits, indents = self.bits, self.indents
|
local bits, indents = self.bits, self.indents
|
||||||
local match = string.match
|
local match = string.match
|
||||||
for i = 1, n do
|
for i = 1, n do
|
||||||
|
local _continue_0 = false
|
||||||
|
repeat
|
||||||
local b = select(i, ...)
|
local b = select(i, ...)
|
||||||
assert(b)
|
assert(b)
|
||||||
|
if b == '' then
|
||||||
|
_continue_0 = true
|
||||||
|
break
|
||||||
|
end
|
||||||
bits[#bits + 1] = b
|
bits[#bits + 1] = b
|
||||||
if type(b) == 'string' then
|
if type(b) == 'string' then
|
||||||
do
|
do
|
||||||
@ -95,6 +101,11 @@ do
|
|||||||
elseif self.current_indent ~= 0 then
|
elseif self.current_indent ~= 0 then
|
||||||
indents[#bits] = self.current_indent
|
indents[#bits] = self.current_indent
|
||||||
end
|
end
|
||||||
|
_continue_0 = true
|
||||||
|
until true
|
||||||
|
if not _continue_0 then
|
||||||
|
break
|
||||||
|
end
|
||||||
end
|
end
|
||||||
self.__str = nil
|
self.__str = nil
|
||||||
end,
|
end,
|
||||||
|
@ -58,6 +58,7 @@ class Code
|
|||||||
for i=1,n
|
for i=1,n
|
||||||
b = select(i, ...)
|
b = select(i, ...)
|
||||||
assert(b)
|
assert(b)
|
||||||
|
if b == '' then continue
|
||||||
bits[#bits+1] = b
|
bits[#bits+1] = b
|
||||||
if type(b) == 'string'
|
if type(b) == 'string'
|
||||||
if spaces = match(b, "\n([ ]*)[^\n]*$")
|
if spaces = match(b, "\n([ ]*)[^\n]*$")
|
||||||
|
12
files.lua
12
files.lua
@ -12,6 +12,11 @@ files.read = function(filename)
|
|||||||
return file_contents
|
return file_contents
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
if filename == 'stdin' then
|
||||||
|
local contents = io.read('*a')
|
||||||
|
_FILE_CACHE['stdin'] = contents
|
||||||
|
return contents
|
||||||
|
end
|
||||||
local file = io.open(filename)
|
local file = io.open(filename)
|
||||||
if package.nomsupath and not file then
|
if package.nomsupath and not file then
|
||||||
for nomsupath in package.nomsupath:gmatch("[^;]+") do
|
for nomsupath in package.nomsupath:gmatch("[^;]+") do
|
||||||
@ -52,6 +57,9 @@ end
|
|||||||
local ok, lfs = pcall(require, "lfs")
|
local ok, lfs = pcall(require, "lfs")
|
||||||
if ok then
|
if ok then
|
||||||
files.walk = function(path)
|
files.walk = function(path)
|
||||||
|
if match(path, "%.nom$") or match(path, "%.lua$") or path == 'stdin' then
|
||||||
|
return iterate_single, path
|
||||||
|
end
|
||||||
local browse
|
local browse
|
||||||
browse = function(filename)
|
browse = function(filename)
|
||||||
local file_type = lfs.attributes(filename, 'mode')
|
local file_type = lfs.attributes(filename, 'mode')
|
||||||
@ -89,7 +97,7 @@ else
|
|||||||
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)
|
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
|
end
|
||||||
files.walk = function(path)
|
files.walk = function(path)
|
||||||
if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") then
|
if match(path, "%.nom$") or match(path, "%.lua$") or path == 'stdin' then
|
||||||
return iterate_single, path
|
return iterate_single, path
|
||||||
end
|
end
|
||||||
path = gsub(path, "\\", "\\\\")
|
path = gsub(path, "\\", "\\\\")
|
||||||
@ -163,7 +171,7 @@ files.get_line_number = function(str, pos)
|
|||||||
end
|
end
|
||||||
files.get_line = function(str, line_no)
|
files.get_line = function(str, line_no)
|
||||||
local line_starts = files.get_line_starts(str)
|
local line_starts = files.get_line_starts(str)
|
||||||
return str:sub(line_starts[line_no] or 1, line_starts[line_no + 1] or -1)
|
return str:sub(line_starts[line_no] or 1, (line_starts[line_no + 1] or 1) - 2)
|
||||||
end
|
end
|
||||||
files.get_lines = function(str)
|
files.get_lines = function(str)
|
||||||
return get_lines:match(str)
|
return get_lines:match(str)
|
||||||
|
12
files.moon
12
files.moon
@ -13,6 +13,10 @@ files.spoof = (filename, contents)->
|
|||||||
files.read = (filename)->
|
files.read = (filename)->
|
||||||
if file_contents = _FILE_CACHE[filename]
|
if file_contents = _FILE_CACHE[filename]
|
||||||
return file_contents
|
return file_contents
|
||||||
|
if filename == 'stdin'
|
||||||
|
contents = io.read('*a')
|
||||||
|
_FILE_CACHE['stdin'] = contents
|
||||||
|
return contents
|
||||||
file = io.open(filename)
|
file = io.open(filename)
|
||||||
if package.nomsupath and not file
|
if package.nomsupath and not file
|
||||||
for nomsupath in package.nomsupath\gmatch("[^;]+")
|
for nomsupath in package.nomsupath\gmatch("[^;]+")
|
||||||
@ -32,6 +36,8 @@ iterate_single = (item, prev) -> if item == prev then nil else item
|
|||||||
ok, lfs = pcall(require, "lfs")
|
ok, lfs = pcall(require, "lfs")
|
||||||
if ok
|
if ok
|
||||||
files.walk = (path)->
|
files.walk = (path)->
|
||||||
|
if match(path, "%.nom$") or match(path, "%.lua$") or path == 'stdin'
|
||||||
|
return iterate_single, path
|
||||||
-- Return 'true' if any files or directories are found, otherwise 'false'
|
-- Return 'true' if any files or directories are found, otherwise 'false'
|
||||||
browse = (filename)->
|
browse = (filename)->
|
||||||
file_type = lfs.attributes(filename, 'mode')
|
file_type = lfs.attributes(filename, 'mode')
|
||||||
@ -59,9 +65,9 @@ else
|
|||||||
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
|
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
|
||||||
|
|
||||||
files.walk = (path)->
|
files.walk = (path)->
|
||||||
-- Sanitize path
|
if match(path, "%.nom$") or match(path, "%.lua$") or path == 'stdin'
|
||||||
if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$")
|
|
||||||
return iterate_single, path
|
return iterate_single, path
|
||||||
|
-- Sanitize path
|
||||||
-- TODO: improve sanitization
|
-- TODO: improve sanitization
|
||||||
path = gsub(path,"\\","\\\\")
|
path = gsub(path,"\\","\\\\")
|
||||||
path = gsub(path,"`","")
|
path = gsub(path,"`","")
|
||||||
@ -119,7 +125,7 @@ files.get_line_number = (str, pos)->
|
|||||||
|
|
||||||
files.get_line = (str, line_no)->
|
files.get_line = (str, line_no)->
|
||||||
line_starts = files.get_line_starts(str)
|
line_starts = files.get_line_starts(str)
|
||||||
return str\sub(line_starts[line_no] or 1, line_starts[line_no+1] or -1)
|
return str\sub(line_starts[line_no] or 1, (line_starts[line_no+1] or 1) - 2)
|
||||||
|
|
||||||
files.get_lines = (str)-> get_lines\match(str)
|
files.get_lines = (str)-> get_lines\match(str)
|
||||||
|
|
||||||
|
20
nomsu.lua
20
nomsu.lua
@ -85,7 +85,6 @@ do
|
|||||||
end
|
end
|
||||||
local repr
|
local repr
|
||||||
repr = require("utils").repr
|
repr = require("utils").repr
|
||||||
local STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2"
|
|
||||||
if not arg or debug.getinfo(2).func == require then
|
if not arg or debug.getinfo(2).func == require then
|
||||||
return NomsuCompiler
|
return NomsuCompiler
|
||||||
end
|
end
|
||||||
@ -137,7 +136,7 @@ local run
|
|||||||
run = function()
|
run = function()
|
||||||
for i, input in ipairs(args.inputs) do
|
for i, input in ipairs(args.inputs) do
|
||||||
if input == "-" then
|
if input == "-" then
|
||||||
args.inputs[i] = STDIN
|
args.inputs[i] = 'stdin'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if #args.inputs == 0 and not args.interactive then
|
if #args.inputs == 0 and not args.interactive then
|
||||||
@ -173,7 +172,15 @@ run = function()
|
|||||||
local to_run = { }
|
local to_run = { }
|
||||||
local _list_0 = args.inputs
|
local _list_0 = args.inputs
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
|
local _continue_0 = false
|
||||||
|
repeat
|
||||||
local input = _list_0[_index_0]
|
local input = _list_0[_index_0]
|
||||||
|
if input == 'stdin' then
|
||||||
|
input_files[#input_files + 1] = 'stdin'
|
||||||
|
to_run['stdin'] = true
|
||||||
|
_continue_0 = true
|
||||||
|
break
|
||||||
|
end
|
||||||
local found = false
|
local found = false
|
||||||
for f in files.walk(input) do
|
for f in files.walk(input) do
|
||||||
input_files[#input_files + 1] = f
|
input_files[#input_files + 1] = f
|
||||||
@ -183,6 +190,11 @@ run = function()
|
|||||||
if not found then
|
if not found then
|
||||||
error("Could not find: " .. tostring(input))
|
error("Could not find: " .. tostring(input))
|
||||||
end
|
end
|
||||||
|
_continue_0 = true
|
||||||
|
until true
|
||||||
|
if not _continue_0 then
|
||||||
|
break
|
||||||
|
end
|
||||||
end
|
end
|
||||||
nomsu.can_optimize = function(f)
|
nomsu.can_optimize = function(f)
|
||||||
if not (args.optimized) then
|
if not (args.optimized) then
|
||||||
@ -201,8 +213,8 @@ run = function()
|
|||||||
local _continue_0 = false
|
local _continue_0 = false
|
||||||
repeat
|
repeat
|
||||||
local file, source
|
local file, source
|
||||||
if filename == STDIN then
|
if filename == 'stdin' then
|
||||||
file = io.input():read("*a")
|
file = io.read("*a")
|
||||||
files.spoof('stdin', file)
|
files.spoof('stdin', file)
|
||||||
source = Source('stdin', 1, #file)
|
source = Source('stdin', 1, #file)
|
||||||
elseif filename:match("%.nom$") then
|
elseif filename:match("%.nom$") then
|
||||||
|
11
nomsu.moon
11
nomsu.moon
@ -38,7 +38,6 @@ Errhand = require "error_handling"
|
|||||||
NomsuCompiler = require "nomsu_compiler"
|
NomsuCompiler = require "nomsu_compiler"
|
||||||
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
|
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
|
||||||
{:repr} = require "utils"
|
{:repr} = require "utils"
|
||||||
STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2"
|
|
||||||
|
|
||||||
-- If this file was reached via require(), then just return the Nomsu compiler
|
-- If this file was reached via require(), then just return the Nomsu compiler
|
||||||
if not arg or debug.getinfo(2).func == require
|
if not arg or debug.getinfo(2).func == require
|
||||||
@ -89,7 +88,7 @@ FILE_CACHE = setmetatable {}, {
|
|||||||
|
|
||||||
run = ->
|
run = ->
|
||||||
for i,input in ipairs args.inputs
|
for i,input in ipairs args.inputs
|
||||||
if input == "-" then args.inputs[i] = STDIN
|
if input == "-" then args.inputs[i] = 'stdin'
|
||||||
|
|
||||||
if #args.inputs == 0 and not args.interactive
|
if #args.inputs == 0 and not args.interactive
|
||||||
args.inputs = {"core"}
|
args.inputs = {"core"}
|
||||||
@ -114,6 +113,10 @@ run = ->
|
|||||||
input_files = {}
|
input_files = {}
|
||||||
to_run = {}
|
to_run = {}
|
||||||
for input in *args.inputs
|
for input in *args.inputs
|
||||||
|
if input == 'stdin'
|
||||||
|
input_files[#input_files+1] = 'stdin'
|
||||||
|
to_run['stdin'] = true
|
||||||
|
continue
|
||||||
found = false
|
found = false
|
||||||
for f in files.walk(input)
|
for f in files.walk(input)
|
||||||
input_files[#input_files+1] = f
|
input_files[#input_files+1] = f
|
||||||
@ -131,8 +134,8 @@ run = ->
|
|||||||
for arg in *args.inputs
|
for arg in *args.inputs
|
||||||
for filename in files.walk(arg)
|
for filename in files.walk(arg)
|
||||||
local file, source
|
local file, source
|
||||||
if filename == STDIN
|
if filename == 'stdin'
|
||||||
file = io.input!\read("*a")
|
file = io.read("*a")
|
||||||
files.spoof('stdin', file)
|
files.spoof('stdin', file)
|
||||||
source = Source('stdin', 1, #file)
|
source = Source('stdin', 1, #file)
|
||||||
elseif filename\match("%.nom$")
|
elseif filename\match("%.nom$")
|
||||||
|
@ -149,8 +149,8 @@ inline_dict_entry(DictEntry):
|
|||||||
dict_key:
|
dict_key:
|
||||||
text_word / inline_expression
|
text_word / inline_expression
|
||||||
|
|
||||||
comment: "#" [^%nl]* (%nl+ %indent [^%nl]* (%nl+ %nodent [^%nl]*)* %dedent)?
|
comment: ("#" {} {~[^%nl]* (%nl+ (%indent -> '') [^%nl]* (%nl+ (%nodent -> '') [^%nl]*)* %dedent)?~} %userdata) => Comment
|
||||||
eol_comment: "#" [^%nl]*
|
eol_comment: ("#" {} {[^%nl]*} %userdata) => Comment
|
||||||
|
|
||||||
eol: %ws* eol_comment? (!. / &%nl)
|
eol: %ws* eol_comment? (!. / &%nl)
|
||||||
ignored_line: (%nodent comment) / (%ws* (!. / &%nl))
|
ignored_line: (%nodent comment) / (%ws* (!. / &%nl))
|
||||||
|
@ -588,7 +588,7 @@ do
|
|||||||
end
|
end
|
||||||
local bit_lua = self:compile(bit)
|
local bit_lua = self:compile(bit)
|
||||||
if not (bit_lua.is_value) then
|
if not (bit_lua.is_value) then
|
||||||
local src = ' ' .. gsub(tostring(self:tree_to_nomsu(bit)), '\n', '\n ')
|
local src = ' ' .. gsub(tostring(recurse(bit)), '\n', '\n ')
|
||||||
local line = tostring(bit.source.filename) .. ":" .. tostring(files.get_line_number(files.read(bit.source.filename), bit.source.start))
|
local line = tostring(bit.source.filename) .. ":" .. tostring(files.get_line_number(files.read(bit.source.filename), bit.source.start))
|
||||||
self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.")
|
self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.")
|
||||||
end
|
end
|
||||||
@ -698,29 +698,61 @@ do
|
|||||||
return error("Unknown type: " .. tostring(tree.type))
|
return error("Unknown type: " .. tostring(tree.type))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
NomsuCompiler.tree_to_nomsu = function(self, tree, inline, can_use_colon)
|
NomsuCompiler.tree_to_nomsu = function(self, tree, options)
|
||||||
if inline == nil then
|
options = options or { }
|
||||||
inline = false
|
local comments = options.comments
|
||||||
|
if comments == nil and tree.comments then
|
||||||
|
do
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for p, c in pairs(tree.comments) do
|
||||||
|
_accum_0[_len_0] = {
|
||||||
|
comment = c,
|
||||||
|
pos = p
|
||||||
|
}
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
if can_use_colon == nil then
|
comments = _accum_0
|
||||||
can_use_colon = false
|
|
||||||
end
|
end
|
||||||
|
table.sort(comments, function(a, b)
|
||||||
|
return a.pos > b.pos
|
||||||
|
end)
|
||||||
|
end
|
||||||
|
local recurse
|
||||||
|
recurse = function(t, opts)
|
||||||
|
opts = opts or { }
|
||||||
|
opts.comments = comments
|
||||||
|
return self:tree_to_nomsu(t, opts)
|
||||||
|
end
|
||||||
|
local pop_comments
|
||||||
|
pop_comments = function(pos)
|
||||||
|
if not (comments) then
|
||||||
|
return ''
|
||||||
|
end
|
||||||
|
local nomsu = NomsuCode(tree.source)
|
||||||
|
while #comments > 0 and comments[#comments].pos <= pos do
|
||||||
|
local comment = table.remove(comments)
|
||||||
|
nomsu:append("#" .. (gsub(comment.comment, "\n", "\n ")) .. "\n")
|
||||||
|
end
|
||||||
|
if #nomsu.bits == 0 then
|
||||||
|
return ''
|
||||||
|
end
|
||||||
|
return nomsu
|
||||||
|
end
|
||||||
|
local inline, can_use_colon = options.inline, options.can_use_colon
|
||||||
local _exp_0 = tree.type
|
local _exp_0 = tree.type
|
||||||
if "FileChunks" == _exp_0 then
|
if "FileChunks" == _exp_0 then
|
||||||
if inline then
|
if inline then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
local nomsu = NomsuCode(tree.source)
|
local nomsu = NomsuCode(tree.source)
|
||||||
nomsu:concat_append((function()
|
for i, chunk in ipairs(tree) do
|
||||||
local _accum_0 = { }
|
if i > 1 then
|
||||||
local _len_0 = 1
|
nomsu:append("\n\n" .. tostring(("~"):rep(80)) .. "\n\n")
|
||||||
for _index_0 = 1, #tree do
|
end
|
||||||
local c = tree[_index_0]
|
nomsu:append(pop_comments(chunk.source.start))
|
||||||
_accum_0[_len_0] = self:tree_to_nomsu(c)
|
nomsu:append(recurse(chunk))
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
end
|
||||||
return _accum_0
|
|
||||||
end)(), "\n" .. tostring(("~"):rep(80)) .. "\n")
|
|
||||||
return nomsu
|
return nomsu
|
||||||
elseif "Action" == _exp_0 then
|
elseif "Action" == _exp_0 then
|
||||||
if inline then
|
if inline then
|
||||||
@ -732,7 +764,9 @@ do
|
|||||||
end
|
end
|
||||||
nomsu:append(bit)
|
nomsu:append(bit)
|
||||||
else
|
else
|
||||||
local arg_nomsu = self:tree_to_nomsu(bit, true)
|
local arg_nomsu = recurse(bit, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if not (arg_nomsu) then
|
if not (arg_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -762,7 +796,9 @@ do
|
|||||||
elseif bit.type == "Block" then
|
elseif bit.type == "Block" then
|
||||||
arg_nomsu = nil
|
arg_nomsu = nil
|
||||||
else
|
else
|
||||||
arg_nomsu = self:tree_to_nomsu(bit, true)
|
arg_nomsu = recurse(bit, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
end
|
end
|
||||||
if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE then
|
if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE then
|
||||||
if bit.type == "Action" then
|
if bit.type == "Action" then
|
||||||
@ -782,15 +818,17 @@ do
|
|||||||
next_space = " "
|
next_space = " "
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
arg_nomsu = self:tree_to_nomsu(bit, nil, true)
|
arg_nomsu = recurse(bit, {
|
||||||
if not (nomsu) then
|
can_use_colon = true
|
||||||
|
})
|
||||||
|
if not (arg_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
||||||
if i == 1 then
|
if i == 1 then
|
||||||
arg_nomsu = NomsuCode(bit.source, "(..)\n ", arg_nomsu)
|
arg_nomsu = NomsuCode(bit.source, "(..)\n ", pop_comments(bit.source.start), arg_nomsu)
|
||||||
else
|
else
|
||||||
arg_nomsu = NomsuCode(bit.source, "\n ", arg_nomsu)
|
arg_nomsu = NomsuCode(bit.source, "\n ", pop_comments(bit.source.start), arg_nomsu)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then
|
if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then
|
||||||
@ -808,10 +846,12 @@ do
|
|||||||
return nomsu
|
return nomsu
|
||||||
end
|
end
|
||||||
elseif "EscapedNomsu" == _exp_0 then
|
elseif "EscapedNomsu" == _exp_0 then
|
||||||
local nomsu = self:tree_to_nomsu(tree[1], true)
|
local nomsu = recurse(tree[1], {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if nomsu == nil and not inline then
|
if nomsu == nil and not inline then
|
||||||
nomsu = self:tree_to_nomsu(tree[1])
|
nomsu = recurse(tree[1])
|
||||||
return nomsu and NomsuCode(tree.source, "\\:\n ", nomsu)
|
return nomsu and NomsuCode(tree.source, "\\:\n ", pop_comments(tree.source.start), nomsu)
|
||||||
end
|
end
|
||||||
return nomsu and NomsuCode(tree.source, "\\(", nomsu, ")")
|
return nomsu and NomsuCode(tree.source, "\\(", nomsu, ")")
|
||||||
elseif "Block" == _exp_0 then
|
elseif "Block" == _exp_0 then
|
||||||
@ -821,7 +861,9 @@ do
|
|||||||
if i > 1 then
|
if i > 1 then
|
||||||
nomsu:append("; ")
|
nomsu:append("; ")
|
||||||
end
|
end
|
||||||
local line_nomsu = self:tree_to_nomsu(line, true)
|
local line_nomsu = recurse(line, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if not (line_nomsu) then
|
if not (line_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -831,7 +873,10 @@ do
|
|||||||
end
|
end
|
||||||
local nomsu = NomsuCode(tree.source)
|
local nomsu = NomsuCode(tree.source)
|
||||||
for i, line in ipairs(tree) do
|
for i, line in ipairs(tree) do
|
||||||
line = assert(self:tree_to_nomsu(line, nil, true), "Could not convert line to nomsu")
|
nomsu:append(pop_comments(line.source.start))
|
||||||
|
line = assert(recurse(line, {
|
||||||
|
can_use_colon = true
|
||||||
|
}), "Could not convert line to nomsu")
|
||||||
nomsu:append(line)
|
nomsu:append(line)
|
||||||
if i < #tree then
|
if i < #tree then
|
||||||
nomsu:append("\n")
|
nomsu:append("\n")
|
||||||
@ -849,7 +894,9 @@ do
|
|||||||
if type(bit) == 'string' then
|
if type(bit) == 'string' then
|
||||||
nomsu:append((gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"')))
|
nomsu:append((gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"')))
|
||||||
else
|
else
|
||||||
local interp_nomsu = self:tree_to_nomsu(bit, true)
|
local interp_nomsu = recurse(bit, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if interp_nomsu then
|
if interp_nomsu then
|
||||||
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
||||||
interp_nomsu:parenthesize()
|
interp_nomsu:parenthesize()
|
||||||
@ -863,7 +910,9 @@ do
|
|||||||
nomsu:append('"')
|
nomsu:append('"')
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
local inline_version = self:tree_to_nomsu(tree, true)
|
local inline_version = recurse(tree, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if inline_version and #inline_version <= MAX_LINE then
|
if inline_version and #inline_version <= MAX_LINE then
|
||||||
return inline_version
|
return inline_version
|
||||||
end
|
end
|
||||||
@ -901,14 +950,16 @@ do
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
local interp_nomsu = self:tree_to_nomsu(bit, true)
|
local interp_nomsu = recurse(bit, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if interp_nomsu then
|
if interp_nomsu then
|
||||||
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
||||||
interp_nomsu:parenthesize()
|
interp_nomsu:parenthesize()
|
||||||
end
|
end
|
||||||
nomsu:append("\\", interp_nomsu)
|
nomsu:append("\\", interp_nomsu)
|
||||||
else
|
else
|
||||||
interp_nomsu = assert(self:tree_to_nomsu(bit))
|
interp_nomsu = assert(recurse(bit))
|
||||||
if not (interp_nomsu) then
|
if not (interp_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -925,7 +976,9 @@ do
|
|||||||
if inline then
|
if inline then
|
||||||
local nomsu = NomsuCode(tree.source, "[")
|
local nomsu = NomsuCode(tree.source, "[")
|
||||||
for i, item in ipairs(tree) do
|
for i, item in ipairs(tree) do
|
||||||
local item_nomsu = self:tree_to_nomsu(item, true)
|
local item_nomsu = recurse(item, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if not (item_nomsu) then
|
if not (item_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -937,30 +990,47 @@ do
|
|||||||
nomsu:append("]")
|
nomsu:append("]")
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
local inline_version = self:tree_to_nomsu(tree, true)
|
local inline_version = recurse(tree, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if inline_version and #inline_version <= MAX_LINE then
|
if inline_version and #inline_version <= MAX_LINE then
|
||||||
return inline_version
|
return inline_version
|
||||||
end
|
end
|
||||||
local nomsu = NomsuCode(tree.source, "[..]")
|
local nomsu = NomsuCode(tree.source, "[..]")
|
||||||
local line = NomsuCode(tree.source, "\n ")
|
local line = NomsuCode(tree.source, "\n ")
|
||||||
for _index_0 = 1, #tree do
|
local line_comments
|
||||||
local item = tree[_index_0]
|
if #tree > 0 then
|
||||||
local item_nomsu = self:tree_to_nomsu(item, true)
|
line_comments = pop_comments(tree[1].source.start)
|
||||||
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then
|
else
|
||||||
|
line_comments = ''
|
||||||
|
end
|
||||||
|
for i, item in ipairs(tree) do
|
||||||
|
local item_nomsu = recurse(item, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
|
if item_nomsu and #tostring(line) + #", " + #item_nomsu <= MAX_LINE then
|
||||||
if #line.bits > 1 then
|
if #line.bits > 1 then
|
||||||
line:append(", ")
|
line:append(", ")
|
||||||
end
|
end
|
||||||
line:append(item_nomsu)
|
line:append(item_nomsu)
|
||||||
else
|
else
|
||||||
if not (item_nomsu) then
|
if not (item_nomsu) then
|
||||||
item_nomsu = self:tree_to_nomsu(item)
|
item_nomsu = recurse(item)
|
||||||
if not (item_nomsu) then
|
if not (item_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if #line.bits > 1 then
|
if #line.bits > 1 then
|
||||||
|
if #tostring(line_comments) > 0 then
|
||||||
|
nomsu:append('\n ', line_comments)
|
||||||
|
end
|
||||||
nomsu:append(line)
|
nomsu:append(line)
|
||||||
line = NomsuCode(line.source, "\n ")
|
line = NomsuCode(line.source, "\n ")
|
||||||
|
if i < #tree then
|
||||||
|
line_comments = pop_comments(tree[i + 1].source.start)
|
||||||
|
else
|
||||||
|
line_comments = ''
|
||||||
|
end
|
||||||
end
|
end
|
||||||
line:append(item_nomsu)
|
line:append(item_nomsu)
|
||||||
end
|
end
|
||||||
@ -974,7 +1044,9 @@ do
|
|||||||
if inline then
|
if inline then
|
||||||
local nomsu = NomsuCode(tree.source, "{")
|
local nomsu = NomsuCode(tree.source, "{")
|
||||||
for i, entry in ipairs(tree) do
|
for i, entry in ipairs(tree) do
|
||||||
local entry_nomsu = self:tree_to_nomsu(entry, true)
|
local entry_nomsu = recurse(entry, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if not (entry_nomsu) then
|
if not (entry_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -986,15 +1058,22 @@ do
|
|||||||
nomsu:append("}")
|
nomsu:append("}")
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
local inline_version = self:tree_to_nomsu(tree, true)
|
local inline_version = recurse(tree, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if inline_version then
|
if inline_version then
|
||||||
return inline_version
|
return inline_version
|
||||||
end
|
end
|
||||||
local nomsu = NomsuCode(tree.source, "{..}")
|
local nomsu = NomsuCode(tree.source, "{..}")
|
||||||
local line = NomsuCode(tree.source, "\n ")
|
local line = NomsuCode(tree.source, "\n ")
|
||||||
for _index_0 = 1, #tree do
|
local line_comments
|
||||||
local entry = tree[_index_0]
|
if #tree > 0 then
|
||||||
local entry_nomsu = self:tree_to_nomsu(entry)
|
line_comments = pop_comments(tree[1].source.start)
|
||||||
|
else
|
||||||
|
line_comments = ''
|
||||||
|
end
|
||||||
|
for i, entry in ipairs(tree) do
|
||||||
|
local entry_nomsu = recurse(entry)
|
||||||
if not (entry_nomsu) then
|
if not (entry_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -1005,8 +1084,16 @@ do
|
|||||||
line:append(entry_nomsu)
|
line:append(entry_nomsu)
|
||||||
else
|
else
|
||||||
if #line.bits > 1 then
|
if #line.bits > 1 then
|
||||||
|
if #tostring(line_comments) > 0 then
|
||||||
|
nomsu:append("\n ", line_comments)
|
||||||
|
end
|
||||||
nomsu:append(line)
|
nomsu:append(line)
|
||||||
line = NomsuCode(line.source, "\n ")
|
line = NomsuCode(line.source, "\n ")
|
||||||
|
if i < #tree then
|
||||||
|
line_comments = pop_comments(tree[1].source.start)
|
||||||
|
else
|
||||||
|
line_comments = ''
|
||||||
|
end
|
||||||
end
|
end
|
||||||
line:append(entry_nomsu)
|
line:append(entry_nomsu)
|
||||||
end
|
end
|
||||||
@ -1018,7 +1105,9 @@ do
|
|||||||
end
|
end
|
||||||
elseif "DictEntry" == _exp_0 then
|
elseif "DictEntry" == _exp_0 then
|
||||||
local key, value = tree[1], tree[2]
|
local key, value = tree[1], tree[2]
|
||||||
local key_nomsu = self:tree_to_nomsu(key, true)
|
local key_nomsu = recurse(key, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
if not (key_nomsu) then
|
if not (key_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -1027,7 +1116,9 @@ do
|
|||||||
end
|
end
|
||||||
local value_nomsu
|
local value_nomsu
|
||||||
if value then
|
if value then
|
||||||
value_nomsu = self:tree_to_nomsu(value, true)
|
value_nomsu = recurse(value, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
else
|
else
|
||||||
value_nomsu = NomsuCode(tree.source, "")
|
value_nomsu = NomsuCode(tree.source, "")
|
||||||
end
|
end
|
||||||
@ -1038,7 +1129,7 @@ do
|
|||||||
if inline then
|
if inline then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
value_nomsu = self:tree_to_nomsu(value)
|
value_nomsu = recurse(value)
|
||||||
if not (value_nomsu) then
|
if not (value_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -1057,7 +1148,9 @@ do
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
if not (bit_nomsu) then
|
if not (bit_nomsu) then
|
||||||
bit_nomsu = self:tree_to_nomsu(bit, true)
|
bit_nomsu = recurse(bit, {
|
||||||
|
inline = true
|
||||||
|
})
|
||||||
end
|
end
|
||||||
if not (bit_nomsu) then
|
if not (bit_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
|
@ -374,7 +374,7 @@ with NomsuCompiler
|
|||||||
string_buffer = ""
|
string_buffer = ""
|
||||||
bit_lua = @compile(bit)
|
bit_lua = @compile(bit)
|
||||||
unless bit_lua.is_value
|
unless bit_lua.is_value
|
||||||
src = ' '..gsub(tostring(@tree_to_nomsu(bit)), '\n','\n ')
|
src = ' '..gsub(tostring(recurse(bit)), '\n','\n ')
|
||||||
line = "#{bit.source.filename}:#{files.get_line_number(files.read(bit.source.filename), bit.source.start)}"
|
line = "#{bit.source.filename}:#{files.get_line_number(files.read(bit.source.filename), bit.source.start)}"
|
||||||
@compile_error bit,
|
@compile_error bit,
|
||||||
"Cannot use:\n%s\nas a string interpolation value, since it's not an expression."
|
"Cannot use:\n%s\nas a string interpolation value, since it's not an expression."
|
||||||
@ -472,12 +472,35 @@ with NomsuCompiler
|
|||||||
else
|
else
|
||||||
error("Unknown type: #{tree.type}")
|
error("Unknown type: #{tree.type}")
|
||||||
|
|
||||||
.tree_to_nomsu = (tree, inline=false, can_use_colon=false)=>
|
.tree_to_nomsu = (tree, options)=>
|
||||||
|
options or= {}
|
||||||
|
comments = options.comments
|
||||||
|
if comments == nil and tree.comments
|
||||||
|
comments = [{comment:c, pos:p} for p,c in pairs tree.comments]
|
||||||
|
table.sort comments, (a,b)-> a.pos > b.pos
|
||||||
|
recurse = (t, opts)->
|
||||||
|
opts or= {}
|
||||||
|
opts.comments = comments
|
||||||
|
return @tree_to_nomsu(t, opts)
|
||||||
|
pop_comments = (pos)->
|
||||||
|
return '' unless comments
|
||||||
|
nomsu = NomsuCode(tree.source)
|
||||||
|
while #comments > 0 and comments[#comments].pos <= pos
|
||||||
|
comment = table.remove comments
|
||||||
|
nomsu\append "#"..(gsub(comment.comment, "\n", "\n ")).."\n"
|
||||||
|
if #nomsu.bits == 0 then return ''
|
||||||
|
return nomsu
|
||||||
|
|
||||||
|
inline, can_use_colon = options.inline, options.can_use_colon
|
||||||
switch tree.type
|
switch tree.type
|
||||||
when "FileChunks"
|
when "FileChunks"
|
||||||
return nil if inline
|
return nil if inline
|
||||||
nomsu = NomsuCode(tree.source)
|
nomsu = NomsuCode(tree.source)
|
||||||
nomsu\concat_append [@tree_to_nomsu(c) for c in *tree], "\n#{("~")\rep(80)}\n"
|
for i, chunk in ipairs tree
|
||||||
|
if i > 1
|
||||||
|
nomsu\append "\n\n#{("~")\rep(80)}\n\n"
|
||||||
|
nomsu\append pop_comments(chunk.source.start)
|
||||||
|
nomsu\append recurse(chunk)
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
when "Action"
|
when "Action"
|
||||||
@ -489,7 +512,7 @@ with NomsuCompiler
|
|||||||
nomsu\append " "
|
nomsu\append " "
|
||||||
nomsu\append bit
|
nomsu\append bit
|
||||||
else
|
else
|
||||||
arg_nomsu = @tree_to_nomsu(bit,true)
|
arg_nomsu = recurse(bit,inline:true)
|
||||||
return nil unless arg_nomsu
|
return nil unless arg_nomsu
|
||||||
unless i == 1
|
unless i == 1
|
||||||
nomsu\append " "
|
nomsu\append " "
|
||||||
@ -509,7 +532,7 @@ with NomsuCompiler
|
|||||||
else
|
else
|
||||||
arg_nomsu = if last_colon == i-1 and bit.type == "Action" then nil
|
arg_nomsu = if last_colon == i-1 and bit.type == "Action" then nil
|
||||||
elseif bit.type == "Block" then nil
|
elseif bit.type == "Block" then nil
|
||||||
else @tree_to_nomsu(bit,true)
|
else recurse(bit,inline:true)
|
||||||
|
|
||||||
if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE
|
if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE
|
||||||
if bit.type == "Action"
|
if bit.type == "Action"
|
||||||
@ -527,14 +550,14 @@ with NomsuCompiler
|
|||||||
line_len += #next_space + #tostring(arg_nomsu)
|
line_len += #next_space + #tostring(arg_nomsu)
|
||||||
next_space = " "
|
next_space = " "
|
||||||
else
|
else
|
||||||
arg_nomsu = @tree_to_nomsu(bit, nil, true)
|
arg_nomsu = recurse(bit, can_use_colon:true)
|
||||||
return nil unless nomsu
|
return nil unless arg_nomsu
|
||||||
-- These types carry their own indentation
|
-- These types carry their own indentation
|
||||||
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
||||||
if i == 1
|
if i == 1
|
||||||
arg_nomsu = NomsuCode(bit.source, "(..)\n ", arg_nomsu)
|
arg_nomsu = NomsuCode(bit.source, "(..)\n ", pop_comments(bit.source.start), arg_nomsu)
|
||||||
else
|
else
|
||||||
arg_nomsu = NomsuCode(bit.source, "\n ", arg_nomsu)
|
arg_nomsu = NomsuCode(bit.source, "\n ", pop_comments(bit.source.start), arg_nomsu)
|
||||||
|
|
||||||
if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block")
|
if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block")
|
||||||
next_space = ""
|
next_space = ""
|
||||||
@ -547,10 +570,10 @@ with NomsuCompiler
|
|||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
when "EscapedNomsu"
|
when "EscapedNomsu"
|
||||||
nomsu = @tree_to_nomsu(tree[1], true)
|
nomsu = recurse(tree[1], inline:true)
|
||||||
if nomsu == nil and not inline
|
if nomsu == nil and not inline
|
||||||
nomsu = @tree_to_nomsu(tree[1])
|
nomsu = recurse(tree[1])
|
||||||
return nomsu and NomsuCode tree.source, "\\:\n ", nomsu
|
return nomsu and NomsuCode tree.source, "\\:\n ", pop_comments(tree.source.start), nomsu
|
||||||
return nomsu and NomsuCode tree.source, "\\(", nomsu, ")"
|
return nomsu and NomsuCode tree.source, "\\(", nomsu, ")"
|
||||||
|
|
||||||
when "Block"
|
when "Block"
|
||||||
@ -559,13 +582,14 @@ with NomsuCompiler
|
|||||||
for i,line in ipairs tree
|
for i,line in ipairs tree
|
||||||
if i > 1
|
if i > 1
|
||||||
nomsu\append "; "
|
nomsu\append "; "
|
||||||
line_nomsu = @tree_to_nomsu(line,true)
|
line_nomsu = recurse(line,inline:true)
|
||||||
return nil unless line_nomsu
|
return nil unless line_nomsu
|
||||||
nomsu\append line_nomsu
|
nomsu\append line_nomsu
|
||||||
return nomsu
|
return nomsu
|
||||||
nomsu = NomsuCode(tree.source)
|
nomsu = NomsuCode(tree.source)
|
||||||
for i, line in ipairs tree
|
for i, line in ipairs tree
|
||||||
line = assert(@tree_to_nomsu(line, nil, true), "Could not convert line to nomsu")
|
nomsu\append pop_comments(line.source.start)
|
||||||
|
line = assert(recurse(line, can_use_colon:true), "Could not convert line to nomsu")
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
if i < #tree
|
if i < #tree
|
||||||
nomsu\append "\n"
|
nomsu\append "\n"
|
||||||
@ -581,7 +605,7 @@ with NomsuCompiler
|
|||||||
-- TODO: unescape better?
|
-- TODO: unescape better?
|
||||||
nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"'))
|
nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"'))
|
||||||
else
|
else
|
||||||
interp_nomsu = @tree_to_nomsu(bit, true)
|
interp_nomsu = recurse(bit, inline:true)
|
||||||
if interp_nomsu
|
if interp_nomsu
|
||||||
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
||||||
interp_nomsu\parenthesize!
|
interp_nomsu\parenthesize!
|
||||||
@ -590,7 +614,7 @@ with NomsuCompiler
|
|||||||
nomsu\append '"'
|
nomsu\append '"'
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
inline_version = @tree_to_nomsu(tree, true)
|
inline_version = recurse(tree, inline:true)
|
||||||
if inline_version and #inline_version <= MAX_LINE
|
if inline_version and #inline_version <= MAX_LINE
|
||||||
return inline_version
|
return inline_version
|
||||||
nomsu = NomsuCode(tree.source, '".."\n ')
|
nomsu = NomsuCode(tree.source, '".."\n ')
|
||||||
@ -617,13 +641,13 @@ with NomsuCompiler
|
|||||||
else
|
else
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
else
|
else
|
||||||
interp_nomsu = @tree_to_nomsu(bit, true)
|
interp_nomsu = recurse(bit, inline:true)
|
||||||
if interp_nomsu
|
if interp_nomsu
|
||||||
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
||||||
interp_nomsu\parenthesize!
|
interp_nomsu\parenthesize!
|
||||||
nomsu\append "\\", interp_nomsu
|
nomsu\append "\\", interp_nomsu
|
||||||
else
|
else
|
||||||
interp_nomsu = assert(@tree_to_nomsu(bit))
|
interp_nomsu = assert(recurse(bit))
|
||||||
return nil unless interp_nomsu
|
return nil unless interp_nomsu
|
||||||
nomsu\append "\\\n ", interp_nomsu
|
nomsu\append "\\\n ", interp_nomsu
|
||||||
if i < #tree
|
if i < #tree
|
||||||
@ -634,7 +658,7 @@ with NomsuCompiler
|
|||||||
if inline
|
if inline
|
||||||
nomsu = NomsuCode(tree.source, "[")
|
nomsu = NomsuCode(tree.source, "[")
|
||||||
for i, item in ipairs tree
|
for i, item in ipairs tree
|
||||||
item_nomsu = @tree_to_nomsu(item, true)
|
item_nomsu = recurse(item, inline:true)
|
||||||
return nil unless item_nomsu
|
return nil unless item_nomsu
|
||||||
if i > 1
|
if i > 1
|
||||||
nomsu\append ", "
|
nomsu\append ", "
|
||||||
@ -642,24 +666,32 @@ with NomsuCompiler
|
|||||||
nomsu\append "]"
|
nomsu\append "]"
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
inline_version = @tree_to_nomsu(tree, true)
|
inline_version = recurse(tree, inline:true)
|
||||||
if inline_version and #inline_version <= MAX_LINE
|
if inline_version and #inline_version <= MAX_LINE
|
||||||
return inline_version
|
return inline_version
|
||||||
nomsu = NomsuCode(tree.source, "[..]")
|
nomsu = NomsuCode(tree.source, "[..]")
|
||||||
line = NomsuCode(tree.source, "\n ")
|
line = NomsuCode(tree.source, "\n ")
|
||||||
for item in *tree
|
line_comments = if #tree > 0
|
||||||
item_nomsu = @tree_to_nomsu(item, true)
|
pop_comments(tree[1].source.start)
|
||||||
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE
|
else ''
|
||||||
|
for i, item in ipairs tree
|
||||||
|
item_nomsu = recurse(item, inline:true)
|
||||||
|
if item_nomsu and #tostring(line) + #", " + #item_nomsu <= MAX_LINE
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
line\append ", "
|
line\append ", "
|
||||||
line\append item_nomsu
|
line\append item_nomsu
|
||||||
else
|
else
|
||||||
unless item_nomsu
|
unless item_nomsu
|
||||||
item_nomsu = @tree_to_nomsu(item)
|
item_nomsu = recurse(item)
|
||||||
return nil unless item_nomsu
|
return nil unless item_nomsu
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
|
if #tostring(line_comments) > 0
|
||||||
|
nomsu\append '\n ', line_comments
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
line = NomsuCode(line.source, "\n ")
|
line = NomsuCode(line.source, "\n ")
|
||||||
|
line_comments = if i < #tree
|
||||||
|
pop_comments(tree[i+1].source.start)
|
||||||
|
else ''
|
||||||
line\append item_nomsu
|
line\append item_nomsu
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
@ -669,7 +701,7 @@ with NomsuCompiler
|
|||||||
if inline
|
if inline
|
||||||
nomsu = NomsuCode(tree.source, "{")
|
nomsu = NomsuCode(tree.source, "{")
|
||||||
for i, entry in ipairs tree
|
for i, entry in ipairs tree
|
||||||
entry_nomsu = @tree_to_nomsu(entry, true)
|
entry_nomsu = recurse(entry, inline:true)
|
||||||
return nil unless entry_nomsu
|
return nil unless entry_nomsu
|
||||||
if i > 1
|
if i > 1
|
||||||
nomsu\append ", "
|
nomsu\append ", "
|
||||||
@ -677,12 +709,15 @@ with NomsuCompiler
|
|||||||
nomsu\append "}"
|
nomsu\append "}"
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
inline_version = @tree_to_nomsu(tree, true)
|
inline_version = recurse(tree, inline:true)
|
||||||
if inline_version then return inline_version
|
if inline_version then return inline_version
|
||||||
nomsu = NomsuCode(tree.source, "{..}")
|
nomsu = NomsuCode(tree.source, "{..}")
|
||||||
line = NomsuCode(tree.source, "\n ")
|
line = NomsuCode(tree.source, "\n ")
|
||||||
for entry in *tree
|
line_comments = if #tree > 0
|
||||||
entry_nomsu = @tree_to_nomsu(entry)
|
pop_comments(tree[1].source.start)
|
||||||
|
else ''
|
||||||
|
for i, entry in ipairs tree
|
||||||
|
entry_nomsu = recurse(entry)
|
||||||
return nil unless entry_nomsu
|
return nil unless entry_nomsu
|
||||||
if #line + #tostring(entry_nomsu) <= MAX_LINE
|
if #line + #tostring(entry_nomsu) <= MAX_LINE
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
@ -690,8 +725,13 @@ with NomsuCompiler
|
|||||||
line\append entry_nomsu
|
line\append entry_nomsu
|
||||||
else
|
else
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
|
if #tostring(line_comments) > 0
|
||||||
|
nomsu\append "\n ", line_comments
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
line = NomsuCode(line.source, "\n ")
|
line = NomsuCode(line.source, "\n ")
|
||||||
|
line_comments = if i < #tree
|
||||||
|
pop_comments(tree[1].source.start)
|
||||||
|
else ''
|
||||||
line\append entry_nomsu
|
line\append entry_nomsu
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
@ -699,17 +739,17 @@ with NomsuCompiler
|
|||||||
|
|
||||||
when "DictEntry"
|
when "DictEntry"
|
||||||
key, value = tree[1], tree[2]
|
key, value = tree[1], tree[2]
|
||||||
key_nomsu = @tree_to_nomsu(key, true)
|
key_nomsu = recurse(key, inline:true)
|
||||||
return nil unless key_nomsu
|
return nil unless key_nomsu
|
||||||
if key.type == "Action" or key.type == "Block"
|
if key.type == "Action" or key.type == "Block"
|
||||||
key_nomsu\parenthesize!
|
key_nomsu\parenthesize!
|
||||||
value_nomsu = if value
|
value_nomsu = if value
|
||||||
@tree_to_nomsu(value, true)
|
recurse(value, inline:true)
|
||||||
else NomsuCode(tree.source, "")
|
else NomsuCode(tree.source, "")
|
||||||
if inline and not value_nomsu then return nil
|
if inline and not value_nomsu then return nil
|
||||||
if not value_nomsu
|
if not value_nomsu
|
||||||
return nil if inline
|
return nil if inline
|
||||||
value_nomsu = @tree_to_nomsu(value)
|
value_nomsu = recurse(value)
|
||||||
return nil unless value_nomsu
|
return nil unless value_nomsu
|
||||||
return NomsuCode tree.source, key_nomsu, ":", value_nomsu
|
return NomsuCode tree.source, key_nomsu, ":", value_nomsu
|
||||||
|
|
||||||
@ -723,7 +763,7 @@ with NomsuCompiler
|
|||||||
-- TODO: support arbitrary words here, including operators and unicode
|
-- TODO: support arbitrary words here, including operators and unicode
|
||||||
if bit[1]\match("[_a-zA-Z][_a-zA-Z0-9]*")
|
if bit[1]\match("[_a-zA-Z][_a-zA-Z0-9]*")
|
||||||
bit_nomsu = bit[1]
|
bit_nomsu = bit[1]
|
||||||
unless bit_nomsu then bit_nomsu = @tree_to_nomsu(bit, true)
|
unless bit_nomsu then bit_nomsu = recurse(bit, inline:true)
|
||||||
return nil unless bit_nomsu
|
return nil unless bit_nomsu
|
||||||
switch bit.type
|
switch bit.type
|
||||||
when "Action", "Block", "IndexChain"
|
when "Action", "Block", "IndexChain"
|
||||||
|
25
parser.lua
25
parser.lua
@ -8,6 +8,7 @@ do
|
|||||||
local _obj_0 = string
|
local _obj_0 = string
|
||||||
match, sub = _obj_0.match, _obj_0.sub
|
match, sub = _obj_0.match, _obj_0.sub
|
||||||
end
|
end
|
||||||
|
local files = require('files')
|
||||||
local NomsuCode, LuaCode, Source
|
local NomsuCode, LuaCode, Source
|
||||||
do
|
do
|
||||||
local _obj_0 = require("code_obj")
|
local _obj_0 = require("code_obj")
|
||||||
@ -74,12 +75,11 @@ do
|
|||||||
return #src + 1
|
return #src + 1
|
||||||
end
|
end
|
||||||
local err_pos = start_pos
|
local err_pos = start_pos
|
||||||
local line_no = pos_to_line(src, err_pos)
|
local line_no = files.get_line_number(src, err_pos)
|
||||||
local line_starts = LINE_STARTS[src]
|
local prev_line = line_no == 1 and "" or files.get_line(src, line_no - 1)
|
||||||
local prev_line = line_no == 1 and "" or src:sub(line_starts[line_no - 1] or 1, line_starts[line_no] - 2)
|
local err_line = files.get_line(src, line_no)
|
||||||
local err_line = src:sub(line_starts[line_no], (line_starts[line_no + 1] or 0) - 2)
|
local next_line = files.get_line(src, line_no + 1)
|
||||||
local next_line = src:sub(line_starts[line_no + 1] or -1, (line_starts[line_no + 2] or 0) - 2)
|
local i = err_pos - files.get_line_starts(src)[line_no]
|
||||||
local i = err_pos - line_starts[line_no]
|
|
||||||
local pointer = ("-"):rep(i) .. "^"
|
local pointer = ("-"):rep(i) .. "^"
|
||||||
err_msg = colored.bright(colored.yellow(colored.onred((err_msg or "Parse error") .. " at " .. tostring(userdata.source.filename) .. ":" .. tostring(line_no) .. ":")))
|
err_msg = colored.bright(colored.yellow(colored.onred((err_msg or "Parse error") .. " at " .. tostring(userdata.source.filename) .. ":" .. tostring(line_no) .. ":")))
|
||||||
if #prev_line > 0 then
|
if #prev_line > 0 then
|
||||||
@ -93,6 +93,10 @@ do
|
|||||||
seen_errors[start_pos] = err_msg
|
seen_errors[start_pos] = err_msg
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
|
_with_0.Comment = function(src, end_pos, start_pos, value, userdata)
|
||||||
|
userdata.comments[start_pos] = value
|
||||||
|
return true
|
||||||
|
end
|
||||||
NOMSU_DEFS = _with_0
|
NOMSU_DEFS = _with_0
|
||||||
end
|
end
|
||||||
setmetatable(NOMSU_DEFS, {
|
setmetatable(NOMSU_DEFS, {
|
||||||
@ -109,9 +113,6 @@ setmetatable(NOMSU_DEFS, {
|
|||||||
if value.__init then
|
if value.__init then
|
||||||
value:__init()
|
value:__init()
|
||||||
end
|
end
|
||||||
for i = 1, #value do
|
|
||||||
assert(value[i])
|
|
||||||
end
|
|
||||||
return value
|
return value
|
||||||
end
|
end
|
||||||
self[key] = make_node
|
self[key] = make_node
|
||||||
@ -158,14 +159,15 @@ Parser.parse = function(nomsu_code, source)
|
|||||||
local userdata = {
|
local userdata = {
|
||||||
indent = "",
|
indent = "",
|
||||||
errors = { },
|
errors = { },
|
||||||
source = source
|
source = source,
|
||||||
|
comments = { }
|
||||||
}
|
}
|
||||||
local tree = NOMSU_PATTERN:match(nomsu_code, nil, userdata)
|
local tree = NOMSU_PATTERN:match(nomsu_code, nil, userdata)
|
||||||
if not (tree) then
|
if not (tree) then
|
||||||
error("In file " .. tostring(colored.blue(tostring(source or "<unknown>"))) .. " failed to parse:\n" .. tostring(colored.onyellow(colored.black(nomsu_code))))
|
error("In file " .. tostring(colored.blue(tostring(source or "<unknown>"))) .. " failed to parse:\n" .. tostring(colored.onyellow(colored.black(nomsu_code))))
|
||||||
end
|
end
|
||||||
if type(tree) == 'number' then
|
if type(tree) == 'number' then
|
||||||
tree = nil
|
return nil
|
||||||
end
|
end
|
||||||
if next(userdata.errors) then
|
if next(userdata.errors) then
|
||||||
local keys
|
local keys
|
||||||
@ -192,6 +194,7 @@ Parser.parse = function(nomsu_code, source)
|
|||||||
end
|
end
|
||||||
error("Errors occurred while parsing:\n\n" .. table.concat(errors, "\n\n"), 0)
|
error("Errors occurred while parsing:\n\n" .. table.concat(errors, "\n\n"), 0)
|
||||||
end
|
end
|
||||||
|
tree.comments = userdata.comments
|
||||||
return tree
|
return tree
|
||||||
end
|
end
|
||||||
return Parser
|
return Parser
|
||||||
|
22
parser.moon
22
parser.moon
@ -4,6 +4,7 @@ re = require 're'
|
|||||||
lpeg.setmaxstack 10000
|
lpeg.setmaxstack 10000
|
||||||
{:P,:R,:S,:C,:Cmt,:Carg} = lpeg
|
{:P,:R,:S,:C,:Cmt,:Carg} = lpeg
|
||||||
{:match, :sub} = string
|
{:match, :sub} = string
|
||||||
|
files = require 'files'
|
||||||
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
|
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
|
||||||
AST = require "nomsu_tree"
|
AST = require "nomsu_tree"
|
||||||
|
|
||||||
@ -56,13 +57,12 @@ NOMSU_DEFS = with {}
|
|||||||
seen_errors[start_pos+1] = colored.bright colored.yellow colored.onred "Too many errors, canceling parsing..."
|
seen_errors[start_pos+1] = colored.bright colored.yellow colored.onred "Too many errors, canceling parsing..."
|
||||||
return #src+1
|
return #src+1
|
||||||
err_pos = start_pos
|
err_pos = start_pos
|
||||||
line_no = pos_to_line(src, err_pos)
|
line_no = files.get_line_number(src, err_pos)
|
||||||
--src = files.read(userdata.source.filename)
|
--src = files.read(userdata.source.filename)
|
||||||
line_starts = LINE_STARTS[src]
|
prev_line = line_no == 1 and "" or files.get_line(src, line_no-1)
|
||||||
prev_line = line_no == 1 and "" or src\sub(line_starts[line_no-1] or 1, line_starts[line_no]-2)
|
err_line = files.get_line(src, line_no)
|
||||||
err_line = src\sub(line_starts[line_no], (line_starts[line_no+1] or 0)-2)
|
next_line = files.get_line(src, line_no+1)
|
||||||
next_line = src\sub(line_starts[line_no+1] or -1, (line_starts[line_no+2] or 0)-2)
|
i = err_pos-files.get_line_starts(src)[line_no]
|
||||||
i = err_pos-line_starts[line_no]
|
|
||||||
pointer = ("-")\rep(i) .. "^"
|
pointer = ("-")\rep(i) .. "^"
|
||||||
err_msg = colored.bright colored.yellow colored.onred (err_msg or "Parse error").." at #{userdata.source.filename}:#{line_no}:"
|
err_msg = colored.bright colored.yellow colored.onred (err_msg or "Parse error").." at #{userdata.source.filename}:#{line_no}:"
|
||||||
if #prev_line > 0 then err_msg ..= "\n"..colored.dim(prev_line)
|
if #prev_line > 0 then err_msg ..= "\n"..colored.dim(prev_line)
|
||||||
@ -72,6 +72,10 @@ NOMSU_DEFS = with {}
|
|||||||
seen_errors[start_pos] = err_msg
|
seen_errors[start_pos] = err_msg
|
||||||
return true
|
return true
|
||||||
|
|
||||||
|
.Comment = (src,end_pos,start_pos,value,userdata)->
|
||||||
|
userdata.comments[start_pos] = value
|
||||||
|
return true
|
||||||
|
|
||||||
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
||||||
make_node = (start, value, stop, userdata)->
|
make_node = (start, value, stop, userdata)->
|
||||||
if userdata.source
|
if userdata.source
|
||||||
@ -79,7 +83,6 @@ setmetatable(NOMSU_DEFS, {__index:(key)=>
|
|||||||
value.source = Source(.filename, .start + start-1, .start + stop-1)
|
value.source = Source(.filename, .start + start-1, .start + stop-1)
|
||||||
setmetatable(value, AST[key])
|
setmetatable(value, AST[key])
|
||||||
if value.__init then value\__init!
|
if value.__init then value\__init!
|
||||||
for i=1,#value do assert(value[i])
|
|
||||||
return value
|
return value
|
||||||
|
|
||||||
self[key] = make_node
|
self[key] = make_node
|
||||||
@ -115,13 +118,13 @@ Parser.parse = (nomsu_code, source=nil)->
|
|||||||
source or= nomsu_code.source
|
source or= nomsu_code.source
|
||||||
nomsu_code = tostring(nomsu_code)
|
nomsu_code = tostring(nomsu_code)
|
||||||
userdata = {
|
userdata = {
|
||||||
indent: "", errors: {}, :source
|
indent: "", errors: {}, :source, comments: {},
|
||||||
}
|
}
|
||||||
tree = NOMSU_PATTERN\match(nomsu_code, nil, userdata)
|
tree = NOMSU_PATTERN\match(nomsu_code, nil, userdata)
|
||||||
unless tree
|
unless tree
|
||||||
error "In file #{colored.blue tostring(source or "<unknown>")} failed to parse:\n#{colored.onyellow colored.black nomsu_code}"
|
error "In file #{colored.blue tostring(source or "<unknown>")} failed to parse:\n#{colored.onyellow colored.black nomsu_code}"
|
||||||
if type(tree) == 'number'
|
if type(tree) == 'number'
|
||||||
tree = nil
|
return nil
|
||||||
|
|
||||||
if next(userdata.errors)
|
if next(userdata.errors)
|
||||||
keys = [k for k,v in pairs(userdata.errors)]
|
keys = [k for k,v in pairs(userdata.errors)]
|
||||||
@ -129,6 +132,7 @@ Parser.parse = (nomsu_code, source=nil)->
|
|||||||
errors = [userdata.errors[k] for k in *keys]
|
errors = [userdata.errors[k] for k in *keys]
|
||||||
error("Errors occurred while parsing:\n\n"..table.concat(errors, "\n\n"), 0)
|
error("Errors occurred while parsing:\n\n"..table.concat(errors, "\n\n"), 0)
|
||||||
|
|
||||||
|
tree.comments = userdata.comments
|
||||||
return tree
|
return tree
|
||||||
|
|
||||||
return Parser
|
return Parser
|
||||||
|
Loading…
Reference in New Issue
Block a user