Fixed some bugs in trailing_line_len() and refactored tree_to_nomsu into

separate inline/not-inline functions.
This commit is contained in:
Bruce Hill 2018-07-19 20:41:31 -07:00
parent 0eff1c77f2
commit 47db74229d
9 changed files with 644 additions and 602 deletions

View File

@ -80,6 +80,11 @@ do
local _class_0 local _class_0
local _base_0 = { local _base_0 = {
is_code = true, is_code = true,
dirty = function(self)
self.__str = nil
self._trailing_line_len = nil
self._is_multiline = nil
end,
append = function(self, ...) append = function(self, ...)
local n = select("#", ...) local n = select("#", ...)
local match = string.match local match = string.match
@ -104,8 +109,7 @@ do
break break
end end
end end
self.__str = nil return self:dirty()
self._trailing_line_len = nil
end, end,
trailing_line_len = function(self) trailing_line_len = function(self)
if self._trailing_line_len == nil then if self._trailing_line_len == nil then
@ -117,19 +121,38 @@ do
do do
local line = match(b, "\n([^\n]*)$") local line = match(b, "\n([^\n]*)$")
if line then if line then
return len + #line len = len + #line
break
else else
len = len + #b len = len + #b
end end
end end
else else
len = len + b:trailing_line_len() len = len + b:trailing_line_len()
if b:is_multiline() then
break
end
end end
end end
self._trailing_line_len = len self._trailing_line_len = len
end end
return self._trailing_line_len return self._trailing_line_len
end, end,
is_multiline = function(self)
if self._is_multiline == nil then
local match = string.match
self._is_multiline = false
local _list_0 = self.bits
for _index_0 = 1, #_list_0 do
local b = _list_0[_index_0]
if type(b) ~= 'string' or match(b, "\n") then
self._is_multiline = true
break
end
end
end
return self._is_multiline
end,
concat_append = function(self, values, joiner, wrapping_joiner) concat_append = function(self, values, joiner, wrapping_joiner)
wrapping_joiner = wrapping_joiner or joiner wrapping_joiner = wrapping_joiner or joiner
local match = string.match local match = string.match
@ -154,8 +177,7 @@ do
line_len = line_len + #b line_len = line_len + #b
end end
end end
self.__str = nil return self:dirty()
self._trailing_line_len = nil
end, end,
prepend = function(self, ...) prepend = function(self, ...)
local n = select("#", ...) local n = select("#", ...)
@ -166,8 +188,7 @@ do
for i = 1, n do for i = 1, n do
bits[i] = select(i, ...) bits[i] = select(i, ...)
end end
self.__str = nil return self:dirty()
self._trailing_line_len = nil
end end
} }
_base_0.__index = _base_0 _base_0.__index = _base_0
@ -223,8 +244,7 @@ do
seen[var] = true seen[var] = true
end end
end end
self.__str = nil return self:dirty()
self._trailing_line_len = nil
end, end,
remove_free_vars = function(self, vars) remove_free_vars = function(self, vars)
if not (#vars > 0) then if not (#vars > 0) then
@ -256,8 +276,7 @@ do
end end
end end
end end
self.__str = nil return self:dirty()
self._trailing_line_len = nil
end, end,
declare_locals = function(self, to_declare) declare_locals = function(self, to_declare)
if to_declare == nil then if to_declare == nil then

View File

@ -51,6 +51,11 @@ class Code
@source = Source\from_string(@source) @source = Source\from_string(@source)
assert(@source and Source\is_instance(@source), "Source has the wrong type") assert(@source and Source\is_instance(@source), "Source has the wrong type")
@append(...) @append(...)
dirty: =>
@__str = nil
@_trailing_line_len = nil
@_is_multiline = nil
append: (...)=> append: (...)=>
n = select("#",...) n = select("#",...)
@ -64,8 +69,7 @@ class Code
bits[#bits+1] = b bits[#bits+1] = b
if type(b) != 'string' and not (type(b) == 'table' and b.is_code) if type(b) != 'string' and not (type(b) == 'table' and b.is_code)
b = repr(b) b = repr(b)
@__str = nil @dirty!
@_trailing_line_len = nil
trailing_line_len: => trailing_line_len: =>
if @_trailing_line_len == nil if @_trailing_line_len == nil
@ -75,13 +79,25 @@ class Code
b = bits[i] b = bits[i]
if type(b) == 'string' if type(b) == 'string'
if line = match(b, "\n([^\n]*)$") if line = match(b, "\n([^\n]*)$")
return len + #line len += #line
break
else len += #b else len += #b
else else
len += b\trailing_line_len! len += b\trailing_line_len!
break if b\is_multiline!
@_trailing_line_len = len @_trailing_line_len = len
return @_trailing_line_len return @_trailing_line_len
is_multiline: =>
if @_is_multiline == nil
match = string.match
@_is_multiline = false
for b in *@bits
if type(b) != 'string' or match(b, "\n")
@_is_multiline = true
break
return @_is_multiline
concat_append: (values, joiner, wrapping_joiner)=> concat_append: (values, joiner, wrapping_joiner)=>
wrapping_joiner or= joiner wrapping_joiner or= joiner
match = string.match match = string.match
@ -102,8 +118,7 @@ class Code
line_len = #line line_len = #line
else else
line_len += #b line_len += #b
@__str = nil @dirty!
@_trailing_line_len = nil
prepend: (...)=> prepend: (...)=>
n = select("#",...) n = select("#",...)
@ -112,8 +127,7 @@ class Code
bits[i] = bits[i-n] bits[i] = bits[i-n]
for i=1,n for i=1,n
bits[i] = select(i, ...) bits[i] = select(i, ...)
@__str = nil @dirty!
@_trailing_line_len = nil
class LuaCode extends Code class LuaCode extends Code
new: (...)=> new: (...)=>
@ -134,8 +148,7 @@ class LuaCode extends Code
unless seen[var] unless seen[var]
@free_vars[#@free_vars+1] = var @free_vars[#@free_vars+1] = var
seen[var] = true seen[var] = true
@__str = nil @dirty!
@_trailing_line_len = nil
remove_free_vars: (vars)=> remove_free_vars: (vars)=>
return unless #vars > 0 return unless #vars > 0
@ -154,8 +167,7 @@ class LuaCode extends Code
for b in *lua.bits for b in *lua.bits
if type(b) != 'string' if type(b) != 'string'
stack[#stack+1] = b stack[#stack+1] = b
@__str = nil @dirty!
@_trailing_line_len = nil
declare_locals: (to_declare=nil)=> declare_locals: (to_declare=nil)=>
if to_declare == nil if to_declare == nil

View File

@ -160,6 +160,9 @@ files.walk = function(path, flush_cache)
end end
local iter local iter
iter = function(_files, i) iter = function(_files, i)
if not (_files) then
return
end
i = i + 1 i = i + 1
do do
local f = _files[i] local f = _files[i]

View File

@ -98,6 +98,7 @@ files.walk = (path, flush_cache=false)->
for nomsupath in package.nomsupath\gmatch("[^;]+") for nomsupath in package.nomsupath\gmatch("[^;]+")
if _files = browse(nomsupath.."/"..path) then break if _files = browse(nomsupath.."/"..path) then break
iter = (_files, i)-> iter = (_files, i)->
return unless _files
i += 1 i += 1
if f = _files[i] if f = _files[i]
return i, f return i, f

View File

@ -733,7 +733,147 @@ do
return error("Unknown type: " .. tostring(tree.type)) return error("Unknown type: " .. tostring(tree.type))
end end
end end
local MIN_COLON_LEN = 20 NomsuCompiler.tree_to_inline_nomsu = function(self, tree, parenthesize_blocks)
if parenthesize_blocks == nil then
parenthesize_blocks = false
end
local _exp_0 = tree.type
if "FileChunks" == _exp_0 then
return error("Cannot inline a FileChunks")
elseif "Action" == _exp_0 then
local nomsu = NomsuCode(tree.source)
for i, bit in ipairs(tree) do
if type(bit) == "string" then
local clump_words = (type(tree[i - 1]) == 'string' and Parser.is_operator(bit) ~= Parser.is_operator(tree[i - 1]))
if i > 1 and not clump_words then
nomsu:append(" ")
end
nomsu:append(bit)
else
local arg_nomsu = self:tree_to_inline_nomsu(bit, parenthesize_blocks or (i == 1 or i < #tree))
if not (tostring(arg_nomsu):match("^:") or i == 1) then
nomsu:append(" ")
end
if bit.type == "Action" then
arg_nomsu:parenthesize()
end
nomsu:append(arg_nomsu)
end
end
return nomsu
elseif "EscapedNomsu" == _exp_0 then
local inner_nomsu = self:tree_to_inline_nomsu(tree[1])
if tree[1].type == "List" or tree[1].type == "Dict" or tree[1].type == "Var" then
return NomsuCode(tree.source, "\\", inner_nomsu)
else
return NomsuCode(tree.source, "\\(", inner_nomsu, ")")
end
elseif "Block" == _exp_0 then
local nomsu = NomsuCode(tree.source, ":")
for i, line in ipairs(tree) do
nomsu:append(i == 1 and " " or "; ")
nomsu:append(self:tree_to_inline_nomsu(line, i == 1 or i < #tree))
end
if #tree > 1 or parenthesize_blocks then
nomsu:parenthesize()
end
return nomsu
elseif "Text" == _exp_0 then
local make_text
make_text = function(tree)
local nomsu = NomsuCode(tree.source)
for i, bit in ipairs(tree) do
if type(bit) == 'string' then
bit = Parser.inline_escape(bit)
nomsu:append(bit)
elseif bit.type == "Text" then
nomsu:append(make_text(bit))
else
local interp_nomsu = self:tree_to_inline_nomsu(bit)
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then
interp_nomsu:parenthesize()
elseif bit.type == "Var" and type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
interp_nomsu:parenthesize()
end
nomsu:append("\\", interp_nomsu)
end
end
return nomsu
end
return NomsuCode(tree.source, '"', make_text(tree), '"')
elseif "List" == _exp_0 then
local nomsu = NomsuCode(tree.source, "[")
for i, item in ipairs(tree) do
if i > 1 then
nomsu:append(", ")
end
nomsu:append(self:tree_to_inline_nomsu(item))
end
nomsu:append("]")
return nomsu
elseif "Dict" == _exp_0 then
local nomsu = NomsuCode(tree.source, "{")
for i, entry in ipairs(tree) do
if i > 1 then
nomsu:append(", ")
end
nomsu:append(self:tree_to_inline_nomsu(entry))
end
nomsu:append("}")
return nomsu
elseif "DictEntry" == _exp_0 then
local key, value = tree[1], tree[2]
local key_nomsu
if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1]) then
key_nomsu = NomsuCode(key.source, key[1])
else
key_nomsu = self:tree_to_inline_nomsu(key)
end
if key.type == "Action" or key.type == "Block" then
key_nomsu:parenthesize()
end
local value_nomsu
if value then
value_nomsu = self:tree_to_inline_nomsu(value)
else
value_nomsu = NomsuCode(tree.source, "")
end
assert(value.type ~= "Block", "Didn't expect to find a Block as a value in a dict")
if value.type == "Block" then
value_nomsu:parenthesize()
end
return NomsuCode(tree.source, key_nomsu, ": ", value_nomsu)
elseif "IndexChain" == _exp_0 then
local nomsu = NomsuCode(tree.source)
for i, bit in ipairs(tree) do
if i > 1 then
nomsu:append(".")
end
local bit_nomsu
if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' and Parser.is_identifier(bit[1]) then
bit_nomsu = bit[1]
else
bit_nomsu = self:tree_to_inline_nomsu(bit)
end
local _exp_1 = bit.type
if "Action" == _exp_1 or "Block" == _exp_1 or "IndexChain" == _exp_1 then
bit_nomsu:parenthesize()
elseif "Number" == _exp_1 then
if i < #tree then
bit_nomsu:parenthesize()
end
end
nomsu:append(bit_nomsu)
end
return nomsu
elseif "Number" == _exp_0 then
return NomsuCode(tree.source, tostring(tree[1]))
elseif "Var" == _exp_0 then
return NomsuCode(tree.source, "%", tree[1])
else
return error("Unknown type: " .. tostring(tree.type))
end
end
NomsuCompiler.tree_to_nomsu = function(self, tree, options) NomsuCompiler.tree_to_nomsu = function(self, tree, options)
options = options or { } options = options or { }
options.consumed_comments = options.consumed_comments or { } options.consumed_comments = options.consumed_comments or { }
@ -787,14 +927,10 @@ do
recurse = function(t, opts) recurse = function(t, opts)
opts = opts or { } opts = opts or { }
opts.consumed_comments = options.consumed_comments opts.consumed_comments = options.consumed_comments
opts.inside_multiblock = opts.inside_multiblock or options.inside_multiblock
return self:tree_to_nomsu(t, opts) return self:tree_to_nomsu(t, opts)
end end
local _exp_0 = tree.type local _exp_0 = tree.type
if "FileChunks" == _exp_0 then if "FileChunks" == _exp_0 then
if options.inline then
error("Cannot inline a FileChunks")
end
local nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) local nomsu = NomsuCode(tree.source, pop_comments(tree.source.start))
for i, chunk in ipairs(tree) do for i, chunk in ipairs(tree) do
if i > 1 then if i > 1 then
@ -808,108 +944,68 @@ do
nomsu:append(pop_comments(tree.source.stop, '\n')) nomsu:append(pop_comments(tree.source.stop, '\n'))
return nomsu return nomsu
elseif "Action" == _exp_0 then elseif "Action" == _exp_0 then
if options.inline then local pos = tree.source.start
local nomsu = NomsuCode(tree.source) local nomsu = NomsuCode(tree.source, pop_comments(pos))
for i, bit in ipairs(tree) do local next_space = ""
if type(bit) == "string" then for i, bit in ipairs(tree) do
if i > 1 and not (type(tree[i - 1]) == 'string' and Parser.is_operator(bit) ~= Parser.is_operator(tree[i - 1])) then if match(next_space, '\n') then
nomsu:append(" ") nomsu:append(pop_comments(pos, '\n'))
end end
nomsu:append(bit) if type(bit) == "string" then
if next_space == ' ' and (type(tree[i - 1]) == 'string' and Parser.is_operator(bit) ~= Parser.is_operator(tree[i - 1])) then
next_space = ''
end
nomsu:append(next_space, bit)
next_space = " "
else
local arg_nomsu
if bit.type == "Block" and #bit > 1 then
arg_nomsu = nil
else else
local arg_nomsu = recurse(bit, { arg_nomsu = self:tree_to_inline_nomsu(bit)
inline = true end
}) if bit.type == "Text" and tostring(arg_nomsu) ~= '"\\n"' and tostring(arg_nomsu):match("\\n") then
if not (arg_nomsu) then arg_nomsu = nil
return nil end
end if bit.type == "Block" then
next_space = match(next_space, "[^ ]*")
end
nomsu:append(next_space)
if arg_nomsu and nomsu:trailing_line_len() + #tostring(arg_nomsu) < MAX_LINE then
if bit.type == "Block" then if bit.type == "Block" then
if i == 1 or i < #tree or options.inside_multiblock then
if i > 1 then
nomsu:append(" ")
end
arg_nomsu:parenthesize()
end
nomsu:append(arg_nomsu) nomsu:append(arg_nomsu)
next_space = "\n.."
else else
if i > 1 then
nomsu:append(" ")
end
if bit.type == "Action" then if bit.type == "Action" then
arg_nomsu:parenthesize() arg_nomsu:parenthesize()
end end
nomsu:append(arg_nomsu) nomsu:append(arg_nomsu)
next_space = " "
end end
end
end
return nomsu
else
local pos = tree.source.start
local nomsu = NomsuCode(tree.source, pop_comments(pos))
local next_space = ""
for i, bit in ipairs(tree) do
if match(next_space, '\n') then
nomsu:append(pop_comments(pos, '\n'))
end
if type(bit) == "string" then
if next_space == ' ' and (type(tree[i - 1]) == 'string' and Parser.is_operator(bit) ~= Parser.is_operator(tree[i - 1])) then
next_space = ''
end
nomsu:append(next_space, bit)
next_space = " "
else else
local arg_nomsu arg_nomsu = assert(recurse(bit))
if bit.type == "Block" and #bit > 1 then local _exp_1 = bit.type
arg_nomsu = nil if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 or "EscapedNomsu" == _exp_1 then
nomsu:append(arg_nomsu)
else else
arg_nomsu = assert(recurse(bit, { nomsu:append(NomsuCode(bit.source, "(..)\n ", pop_comments(bit.source.start), arg_nomsu))
inline = true
}))
end end
if tostring(arg_nomsu) ~= '"\\n"' and tostring(arg_nomsu):match("\\n") then
arg_nomsu = nil
end
if bit.type == "Block" then
next_space = match(next_space, "[^ ]*")
end
nomsu:append(next_space)
if arg_nomsu and nomsu:trailing_line_len() + #tostring(arg_nomsu) < MAX_LINE then
if bit.type == "Block" then
nomsu:append(arg_nomsu)
next_space = "\n.."
else
if bit.type == "Action" then
arg_nomsu:parenthesize()
end
nomsu:append(arg_nomsu)
next_space = " "
end
else
arg_nomsu = assert(recurse(bit))
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" and bit.type ~= "Block" then
nomsu:append(NomsuCode(bit.source, "(..)\n ", pop_comments(bit.source.start), arg_nomsu))
else
nomsu:append(arg_nomsu)
end
next_space = "\n.."
end
pos = bit.source.stop
end
if next_space == " " and nomsu:trailing_line_len() > MAX_LINE then
next_space = "\n.." next_space = "\n.."
end end
pos = bit.source.stop
end
if next_space == " " and nomsu:trailing_line_len() > MAX_LINE then
next_space = "\n.."
end end
nomsu:append(pop_comments(pos, '\n'))
return nomsu
end end
nomsu:append(pop_comments(pos, '\n'))
return nomsu
elseif "EscapedNomsu" == _exp_0 then elseif "EscapedNomsu" == _exp_0 then
local nomsu = NomsuCode(tree.source, "\\(", assert(recurse(tree[1], { local inline_nomsu = self:tree_to_inline_nomsu(tree)
inline = true if #tostring(inline_nomsu) <= MAX_LINE then
})), ")") return inline_nomsu
if options.inline or #tostring(nomsu) <= MAX_LINE then
return nomsu
end end
nomsu = assert(recurse(tree[1])) local nomsu = recurse(tree[1])
local _exp_1 = tree[1].type local _exp_1 = tree[1].type
if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 then if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 then
return NomsuCode(tree.source, "\\", nomsu) return NomsuCode(tree.source, "\\", nomsu)
@ -917,313 +1013,177 @@ do
return NomsuCode(tree.source, "\\(..)\n ", pop_comments(tree.source.start), nomsu) return NomsuCode(tree.source, "\\(..)\n ", pop_comments(tree.source.start), nomsu)
end end
elseif "Block" == _exp_0 then elseif "Block" == _exp_0 then
if options.inline then
local nomsu = NomsuCode(tree.source, ":")
for i, line in ipairs(tree) do
nomsu:append(i == 1 and " " or "; ")
nomsu:append(assert(recurse(line, {
inline = true,
inside_multiblock = true
})))
end
return nomsu
end
local nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) local nomsu = NomsuCode(tree.source, pop_comments(tree.source.start))
for i, line in ipairs(tree) do for i, line in ipairs(tree) do
nomsu:append(pop_comments(line.source.start, '\n')) nomsu:append(pop_comments(line.source.start, '\n'))
line = assert(recurse(line), "Could not convert line to nomsu") local line_nomsu = recurse(line)
nomsu:append(line) nomsu:append(line_nomsu)
if i < #tree then if i < #tree then
nomsu:append("\n") nomsu:append(line_nomsu:is_multiline() and "\n\n" or "\n")
if match(tostring(line), "\n") then
nomsu:append("\n")
end
end end
end end
nomsu:append(pop_comments(tree.source.stop, '\n')) nomsu:append(pop_comments(tree.source.stop, '\n'))
return options.top and nomsu or NomsuCode(tree.source, ":\n ", nomsu) return options.top and nomsu or NomsuCode(tree.source, ":\n ", nomsu)
elseif "Text" == _exp_0 then elseif "Text" == _exp_0 then
if options.inline then local inline_version = self:tree_to_inline_nomsu(tree)
local make_text if inline_version and #tostring(inline_version) <= MAX_LINE then
make_text = function(tree) return inline_version
local nomsu = NomsuCode(tree.source) end
for i, bit in ipairs(tree) do local make_text
if type(bit) == 'string' then make_text = function(tree)
bit = (gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"')) local nomsu = NomsuCode(tree.source)
bit = gsub(bit, "%G", (function(c) for i, bit in ipairs(tree) do
return c == ' ' and c or "\\" .. tostring(c:byte()) if type(bit) == 'string' then
end)) bit = Parser.escape(bit)
nomsu:append(bit) local bit_lines = files.get_lines(bit)
elseif bit.type == "Text" then for j, line in ipairs(bit_lines) do
nomsu:append(make_text(bit)) if j > 1 then
else nomsu:append("\n")
local interp_nomsu = assert(recurse(bit, {
inline = true
}))
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then
interp_nomsu:parenthesize()
elseif bit.type == "Var" and type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
interp_nomsu:parenthesize()
end end
nomsu:append("\\", interp_nomsu) line = gsub(line, "\\", "\\\\")
end if #line > 1.25 * MAX_LINE then
end local remainder = line
return nomsu while #remainder > 0 do
end local split = find(remainder, " ", MAX_LINE, true)
return NomsuCode(tree.source, '"', make_text(tree), '"') if split then
else local chunk
local inline_version = recurse(tree, { chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1)
inline = true nomsu:append(chunk)
}) elseif #remainder > 1.75 * MAX_LINE then
if inline_version and #tostring(inline_version) <= MAX_LINE then split = math.floor(1.5 * MAX_LINE)
return inline_version local chunk
end chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1)
local make_text nomsu:append(chunk)
make_text = function(tree) else
local nomsu = NomsuCode(tree.source) nomsu:append(remainder)
for i, bit in ipairs(tree) do break
if type(bit) == 'string' then end
local bit_lines = files.get_lines(bit) if #remainder > 0 then
for j, line in ipairs(bit_lines) do nomsu:append("\\\n..")
if j > 1 then
nomsu:append("\n")
end
line = gsub(line, "\\", "\\\\")
if #line > 1.25 * MAX_LINE then
local remainder = line
while #remainder > 0 do
local split = find(remainder, " ", MAX_LINE, true)
if split then
local chunk
chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1)
nomsu:append(chunk)
elseif #remainder > 1.75 * MAX_LINE then
split = math.floor(1.5 * MAX_LINE)
local chunk
chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1)
nomsu:append(chunk)
else
nomsu:append(remainder)
break
end
if #remainder > 0 then
nomsu:append("\\\n..")
end
end end
else
nomsu:append(line)
end end
else
nomsu:append(line)
end end
elseif bit.type == "Text" then end
nomsu:append(make_text(bit)) elseif bit.type == "Text" then
else nomsu:append(make_text(bit))
local interp_nomsu = recurse(bit, { else
inline = true local interp_nomsu = self:tree_to_inline_nomsu(bit)
}) if nomsu:trailing_line_len() + #tostring(interp_nomsu) <= MAX_LINE then
if interp_nomsu then local _exp_1 = bit.type
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then if "Var" == _exp_1 then
interp_nomsu:parenthesize() if type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
elseif bit.type == "Var" and type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
interp_nomsu:parenthesize() interp_nomsu:parenthesize()
end end
elseif "List" == _exp_1 or "Dict" == _exp_1 then
nomsu:append("\\", interp_nomsu) nomsu:append("\\", interp_nomsu)
else else
interp_nomsu = assert(recurse(bit)) nomsu:append("\\(", interp_nomsu, ")")
if not (interp_nomsu) then end
return nil else
end interp_nomsu = recurse(bit)
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" and bit.type ~= "Block" then if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" and bit.type ~= "Block" then
nomsu:append("\\\n(..) ", interp_nomsu) nomsu:append("\\(..)\n ", interp_nomsu)
else else
nomsu:append("\\", interp_nomsu) nomsu:append("\\", interp_nomsu)
end end
if i < #tree then if i < #tree then
nomsu:append("\n..") nomsu:append("\n..")
end
end end
end end
end end
return nomsu
end end
return NomsuCode(tree.source, '".."\n ', make_text(tree)) return nomsu
end end
return NomsuCode(tree.source, '".."\n ', make_text(tree))
elseif "List" == _exp_0 then elseif "List" == _exp_0 then
if options.inline then local inline_version = self:tree_to_inline_nomsu(tree)
local nomsu = NomsuCode(tree.source, "[") if inline_version and #tostring(inline_version) <= MAX_LINE then
for i, item in ipairs(tree) do return inline_version
if i > 1 then end
assert(#tree > 0)
local nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start))
for i, item in ipairs(tree) do
local item_nomsu = self:tree_to_inline_nomsu(item)
if item.type == "Block" then
item_nomsu:parenthesize()
end
if nomsu:trailing_line_len() + #tostring(item_nomsu) <= MAX_LINE then
if nomsu:trailing_line_len() > 0 then
nomsu:append(", ") nomsu:append(", ")
end end
nomsu:append(assert(recurse(item, { nomsu:append(item_nomsu)
inline = true else
}))) item_nomsu = recurse(item)
end local _exp_1 = item.type
nomsu:append("]") if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 or "EscapedNomsu" == _exp_1 then
return nomsu
else
local inline_version = recurse(tree, {
inline = true
})
if inline_version and #tostring(inline_version) <= MAX_LINE then
return inline_version
end
assert(#tree > 0)
local nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start))
for i, item in ipairs(tree) do
local item_nomsu = assert(recurse(item, {
inline = true
}))
if item.type == "Block" then
item_nomsu:parenthesize()
end
if nomsu:trailing_line_len() + #tostring(item_nomsu) <= MAX_LINE then
if nomsu:trailing_line_len() > 0 then
nomsu:append(", ")
end
nomsu:append(item_nomsu) nomsu:append(item_nomsu)
else else
if #tostring(item_nomsu) > MAX_LINE then nomsu:append("(..)\n ", item_nomsu)
item_nomsu = recurse(item) end
local _exp_1 = item.type if i < #tree then
if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 then nomsu:append("\n")
nomsu:append(item_nomsu)
else
nomsu:append("(..)\n ", item_nomsu)
end
if i < #tree then
nomsu:append("\n")
end
else
if nomsu:trailing_line_len() > 0 then
nomsu:append('\n')
end
nomsu:append(pop_comments(item.source.start), item_nomsu)
end
end end
end end
nomsu:append(pop_comments(tree.source.stop, '\n'))
return NomsuCode(tree.source, "[..]\n ", nomsu)
end end
nomsu:append(pop_comments(tree.source.stop, '\n'))
return NomsuCode(tree.source, "[..]\n ", nomsu)
elseif "Dict" == _exp_0 then elseif "Dict" == _exp_0 then
if options.inline then local inline_version = self:tree_to_inline_nomsu(tree)
local nomsu = NomsuCode(tree.source, "{") if inline_version and #tostring(inline_version) <= MAX_LINE then
for i, entry in ipairs(tree) do return inline_version
if i > 1 then end
assert(#tree > 0)
local nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start))
for i, item in ipairs(tree) do
local item_nomsu = self:tree_to_inline_nomsu(item)
if item.type == "Block" then
item_nomsu:parenthesize()
end
if nomsu:trailing_line_len() + #tostring(item_nomsu) <= MAX_LINE then
if nomsu:trailing_line_len() > 0 then
nomsu:append(", ") nomsu:append(", ")
end end
nomsu:append(assert(recurse(entry, { nomsu:append(item_nomsu)
inline = true else
}))) item_nomsu = recurse(item)
end local _exp_1 = item.type
nomsu:append("}") if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 or "EscapedNomsu" == _exp_1 then
return nomsu
else
local inline_version = recurse(tree, {
inline = true
})
if inline_version and #tostring(inline_version) <= MAX_LINE then
return inline_version
end
assert(#tree > 0)
local nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start))
for i, item in ipairs(tree) do
local item_nomsu = assert(recurse(item, {
inline = true
}))
if item.type == "Block" then
item_nomsu:parenthesize()
end
if nomsu:trailing_line_len() + #tostring(item_nomsu) <= MAX_LINE then
if nomsu:trailing_line_len() > 0 then
nomsu:append(", ")
end
nomsu:append(item_nomsu) nomsu:append(item_nomsu)
else else
if #tostring(item_nomsu) > MAX_LINE then nomsu:append("(..)\n ", item_nomsu)
item_nomsu = recurse(item) end
local _exp_1 = item.type if i < #tree then
if "List" == _exp_1 or "Dict" == _exp_1 or "Text" == _exp_1 or "Block" == _exp_1 then nomsu:append("\n")
nomsu:append(item_nomsu)
else
nomsu:append("(..)\n ", item_nomsu)
end
if i < #tree then
nomsu:append("\n")
end
else
if nomsu:trailing_line_len() > 0 then
nomsu:append('\n')
end
nomsu:append(pop_comments(item.source.start), item_nomsu)
end
end end
end end
nomsu:append(pop_comments(tree.source.stop, '\n'))
return NomsuCode(tree.source, "{..}\n ", nomsu)
end end
nomsu:append(pop_comments(tree.source.stop, '\n'))
return NomsuCode(tree.source, "{..}\n ", nomsu)
elseif "DictEntry" == _exp_0 then elseif "DictEntry" == _exp_0 then
local inline_version = self:tree_to_inline_nomsu(tree)
if #tostring(inline_version) <= MAX_LINE then
return inline_version
end
local key, value = tree[1], tree[2] local key, value = tree[1], tree[2]
local key_nomsu local key_nomsu
if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1]) then if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1]) then
key_nomsu = NomsuCode(key.source, key[1]) key_nomsu = NomsuCode(key.source, key[1])
else else
key_nomsu = assert(recurse(key, { key_nomsu = self:tree_to_inline_nomsu(key)
inline = true
}))
end end
if key.type == "Action" or key.type == "Block" then if key.type == "Action" or key.type == "Block" then
key_nomsu:parenthesize() key_nomsu:parenthesize()
end end
local value_nomsu local value_nomsu = recurse(value)
if value then if value.type == "List" or value.type == "Dict" or value.type == "Text" or value.type == "Block" then
value_nomsu = assert(recurse(value, {
inline = true
}))
else
value_nomsu = NomsuCode(tree.source, "")
end
assert(value.type ~= "Block", "Didn't expect to find a Block as a value in a dict")
if value.type == "Block" then
value_nomsu:parenthesize()
end
if options.inline or #tostring(key_nomsu) + 2 + #tostring(value_nomsu) <= MAX_LINE then
return NomsuCode(tree.source, key_nomsu, ": ", value_nomsu)
end
value_nomsu = recurse(value)
if value.type == "List" or value.type == "Dict" or value.type == "Text" then
return NomsuCode(tree.source, key_nomsu, ": ", value_nomsu) return NomsuCode(tree.source, key_nomsu, ": ", value_nomsu)
else else
return NomsuCode(tree.source, key_nomsu, ": (..)\n ", value_nomsu) return NomsuCode(tree.source, key_nomsu, ": (..)\n ", value_nomsu)
end end
elseif "IndexChain" == _exp_0 then elseif "IndexChain" == _exp_0 or "Number" == _exp_0 or "Var" == _exp_0 then
local nomsu = NomsuCode(tree.source) return self:tree_to_inline_nomsu(tree)
for i, bit in ipairs(tree) do
if i > 1 then
nomsu:append(".")
end
local bit_nomsu
if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' and bit[1]:match("[_a-zA-Z][_a-zA-Z0-9]*") then
bit_nomsu = bit[1]
else
bit_nomsu = assert(recurse(bit, {
inline = true
}))
end
local _exp_1 = bit.type
if "Action" == _exp_1 or "Block" == _exp_1 or "IndexChain" == _exp_1 then
bit_nomsu:parenthesize()
elseif "Number" == _exp_1 then
if i < #tree then
bit_nomsu:parenthesize()
end
end
nomsu:append(bit_nomsu)
end
return nomsu
elseif "Number" == _exp_0 then
return NomsuCode(tree.source, tostring(tree[1]))
elseif "Var" == _exp_0 then
return NomsuCode(tree.source, "%", tree[1])
else else
return error("Unknown type: " .. tostring(tree.type)) return error("Unknown type: " .. tostring(tree.type))
end end

View File

@ -484,7 +484,114 @@ with NomsuCompiler
else else
error("Unknown type: #{tree.type}") error("Unknown type: #{tree.type}")
MIN_COLON_LEN = 20 -- For beautification purposes, don't bother using colon syntax for short bits .tree_to_inline_nomsu = (tree, parenthesize_blocks=false)=>
switch tree.type
when "FileChunks"
error("Cannot inline a FileChunks")
when "Action"
nomsu = NomsuCode(tree.source)
for i,bit in ipairs tree
if type(bit) == "string"
clump_words = (type(tree[i-1]) == 'string' and Parser.is_operator(bit) != Parser.is_operator(tree[i-1]))
nomsu\append " " if i > 1 and not clump_words
nomsu\append bit
else
arg_nomsu = @tree_to_inline_nomsu(bit, parenthesize_blocks or (i == 1 or i < #tree))
nomsu\append " " unless tostring(arg_nomsu)\match("^:") or i == 1
arg_nomsu\parenthesize! if bit.type == "Action"
nomsu\append arg_nomsu
return nomsu
when "EscapedNomsu"
inner_nomsu = @tree_to_inline_nomsu(tree[1])
if tree[1].type == "List" or tree[1].type == "Dict" or tree[1].type == "Var"
return NomsuCode(tree.source, "\\", inner_nomsu)
else
return NomsuCode(tree.source, "\\(", inner_nomsu, ")")
when "Block"
nomsu = NomsuCode(tree.source, ":")
for i,line in ipairs tree
nomsu\append(i == 1 and " " or "; ")
nomsu\append @tree_to_inline_nomsu(line, i == 1 or i < #tree)
nomsu\parenthesize! if #tree > 1 or parenthesize_blocks
return nomsu
when "Text"
make_text = (tree)->
nomsu = NomsuCode(tree.source)
for i, bit in ipairs tree
if type(bit) == 'string'
bit = Parser.inline_escape(bit)
nomsu\append bit
elseif bit.type == "Text"
nomsu\append(make_text(bit))
else
interp_nomsu = @tree_to_inline_nomsu(bit)
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict"
interp_nomsu\parenthesize!
elseif bit.type == "Var" and type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]")
interp_nomsu\parenthesize!
nomsu\append "\\", interp_nomsu
return nomsu
return NomsuCode(tree.source, '"', make_text(tree), '"')
when "List"
nomsu = NomsuCode(tree.source, "[")
for i, item in ipairs tree
nomsu\append ", " if i > 1
nomsu\append @tree_to_inline_nomsu(item)
nomsu\append "]"
return nomsu
when "Dict"
nomsu = NomsuCode(tree.source, "{")
for i, entry in ipairs tree
nomsu\append ", " if i > 1
nomsu\append @tree_to_inline_nomsu(entry)
nomsu\append "}"
return nomsu
when "DictEntry"
key, value = tree[1], tree[2]
key_nomsu = if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1])
NomsuCode(key.source, key[1])
else @tree_to_inline_nomsu(key)
key_nomsu\parenthesize! if key.type == "Action" or key.type == "Block"
value_nomsu = if value
@tree_to_inline_nomsu(value)
else NomsuCode(tree.source, "")
assert(value.type != "Block", "Didn't expect to find a Block as a value in a dict")
value_nomsu\parenthesize! if value.type == "Block"
return NomsuCode tree.source, key_nomsu, ": ", value_nomsu
when "IndexChain"
nomsu = NomsuCode(tree.source)
for i, bit in ipairs tree
nomsu\append "." if i > 1
local bit_nomsu
bit_nomsu = if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' and Parser.is_identifier(bit[1])
bit[1]
else @tree_to_inline_nomsu(bit)
switch bit.type
when "Action", "Block", "IndexChain"
bit_nomsu\parenthesize!
when "Number"
if i < #tree
bit_nomsu\parenthesize!
nomsu\append bit_nomsu
return nomsu
when "Number"
return NomsuCode(tree.source, tostring(tree[1]))
when "Var"
return NomsuCode(tree.source, "%", tree[1])
else
error("Unknown type: #{tree.type}")
.tree_to_nomsu = (tree, options)=> .tree_to_nomsu = (tree, options)=>
options or= {} options or= {}
options.consumed_comments or= {} options.consumed_comments or= {}
@ -509,12 +616,10 @@ with NomsuCompiler
recurse = (t, opts)-> recurse = (t, opts)->
opts or= {} opts or= {}
opts.consumed_comments = options.consumed_comments opts.consumed_comments = options.consumed_comments
opts.inside_multiblock or= options.inside_multiblock
return @tree_to_nomsu(t, opts) return @tree_to_nomsu(t, opts)
switch tree.type switch tree.type
when "FileChunks" when "FileChunks"
error("Cannot inline a FileChunks") if options.inline
nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) nomsu = NomsuCode(tree.source, pop_comments(tree.source.start))
for i, chunk in ipairs tree for i, chunk in ipairs tree
nomsu\append "\n\n#{("~")\rep(80)}\n\n" if i > 1 nomsu\append "\n\n#{("~")\rep(80)}\n\n" if i > 1
@ -524,72 +629,53 @@ with NomsuCompiler
return nomsu return nomsu
when "Action" when "Action"
if options.inline pos = tree.source.start
nomsu = NomsuCode(tree.source) nomsu = NomsuCode(tree.source, pop_comments(pos))
for i,bit in ipairs tree next_space = ""
if type(bit) == "string" for i,bit in ipairs tree
nomsu\append " " if i > 1 and not (type(tree[i-1]) == 'string' and Parser.is_operator(bit) != Parser.is_operator(tree[i-1])) if match(next_space, '\n')
nomsu\append bit nomsu\append pop_comments(pos, '\n')
else if type(bit) == "string"
arg_nomsu = recurse(bit,inline:true) if next_space == ' ' and (type(tree[i-1]) == 'string' and Parser.is_operator(bit) != Parser.is_operator(tree[i-1]))
return nil unless arg_nomsu next_space = ''
nomsu\append next_space, bit
next_space = " "
else
arg_nomsu = if bit.type == "Block" and #bit > 1 then nil
else @tree_to_inline_nomsu(bit)
if bit.type == "Text" and tostring(arg_nomsu) != '"\\n"' and tostring(arg_nomsu)\match("\\n")
arg_nomsu = nil -- Force long text for multi-line text
next_space = match(next_space, "[^ ]*") if bit.type == "Block"
nomsu\append next_space
if arg_nomsu and nomsu\trailing_line_len! + #tostring(arg_nomsu) < MAX_LINE
if bit.type == "Block" if bit.type == "Block"
if i == 1 or i < #tree or options.inside_multiblock
nomsu\append " " if i > 1
arg_nomsu\parenthesize!
nomsu\append arg_nomsu nomsu\append arg_nomsu
next_space = "\n.."
else else
nomsu\append " " if i > 1
arg_nomsu\parenthesize! if bit.type == "Action" arg_nomsu\parenthesize! if bit.type == "Action"
nomsu\append arg_nomsu nomsu\append arg_nomsu
return nomsu next_space = " "
else
pos = tree.source.start
nomsu = NomsuCode(tree.source, pop_comments(pos))
next_space = ""
for i,bit in ipairs tree
if match(next_space, '\n')
nomsu\append pop_comments(pos, '\n')
if type(bit) == "string"
if next_space == ' ' and (type(tree[i-1]) == 'string' and Parser.is_operator(bit) != Parser.is_operator(tree[i-1]))
next_space = ''
nomsu\append next_space, bit
next_space = " "
else else
arg_nomsu = if bit.type == "Block" and #bit > 1 then nil arg_nomsu = assert recurse(bit)
else assert recurse(bit,inline:true) -- These types carry their own indentation
if tostring(arg_nomsu) != '"\\n"' and tostring(arg_nomsu)\match("\\n") switch bit.type
arg_nomsu = nil -- Force long text for multi-line text when "List", "Dict", "Text", "Block", "EscapedNomsu"
next_space = match(next_space, "[^ ]*") if bit.type == "Block"
nomsu\append next_space
if arg_nomsu and nomsu\trailing_line_len! + #tostring(arg_nomsu) < MAX_LINE
if bit.type == "Block"
nomsu\append arg_nomsu nomsu\append arg_nomsu
next_space = "\n.."
else else
arg_nomsu\parenthesize! if bit.type == "Action"
nomsu\append arg_nomsu
next_space = " "
else
arg_nomsu = assert recurse(bit)
-- These types carry their own indentation
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" and bit.type != "Block"
nomsu\append NomsuCode(bit.source, "(..)\n ", pop_comments(bit.source.start), arg_nomsu) nomsu\append NomsuCode(bit.source, "(..)\n ", pop_comments(bit.source.start), arg_nomsu)
else
nomsu\append arg_nomsu
next_space = "\n.."
pos = bit.source.stop
if next_space == " " and nomsu\trailing_line_len! > MAX_LINE
next_space = "\n.." next_space = "\n.."
nomsu\append pop_comments(pos, '\n') pos = bit.source.stop
return nomsu
if next_space == " " and nomsu\trailing_line_len! > MAX_LINE
next_space = "\n.."
nomsu\append pop_comments(pos, '\n')
return nomsu
when "EscapedNomsu" when "EscapedNomsu"
nomsu = NomsuCode(tree.source, "\\(", assert(recurse(tree[1], inline:true)), ")") inline_nomsu = @tree_to_inline_nomsu tree
if options.inline or #tostring(nomsu) <= MAX_LINE if #tostring(inline_nomsu) <= MAX_LINE
return nomsu return inline_nomsu
nomsu = assert recurse(tree[1]) nomsu = recurse(tree[1])
switch tree[1].type switch tree[1].type
when "List", "Dict", "Text", "Block" when "List", "Dict", "Text", "Block"
return NomsuCode tree.source, "\\", nomsu return NomsuCode tree.source, "\\", nomsu
@ -597,208 +683,133 @@ with NomsuCompiler
return NomsuCode tree.source, "\\(..)\n ", pop_comments(tree.source.start), nomsu return NomsuCode tree.source, "\\(..)\n ", pop_comments(tree.source.start), nomsu
when "Block" when "Block"
if options.inline
nomsu = NomsuCode(tree.source, ":")
for i,line in ipairs tree
nomsu\append(i == 1 and " " or "; ")
nomsu\append assert(recurse(line, inline:true, inside_multiblock:true))
return nomsu
nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) nomsu = NomsuCode(tree.source, pop_comments(tree.source.start))
for i, line in ipairs tree for i, line in ipairs tree
nomsu\append pop_comments(line.source.start, '\n') nomsu\append pop_comments(line.source.start, '\n')
line = assert(recurse(line), "Could not convert line to nomsu") line_nomsu = recurse(line)
nomsu\append line nomsu\append line_nomsu
if i < #tree if i < #tree
nomsu\append "\n" nomsu\append(line_nomsu\is_multiline! and "\n\n" or "\n")
if match(tostring(line), "\n")
nomsu\append "\n"
nomsu\append pop_comments(tree.source.stop, '\n') nomsu\append pop_comments(tree.source.stop, '\n')
return options.top and nomsu or NomsuCode(tree.source, ":\n ", nomsu) return options.top and nomsu or NomsuCode(tree.source, ":\n ", nomsu)
when "Text" when "Text"
if options.inline inline_version = @tree_to_inline_nomsu(tree)
make_text = (tree)-> if inline_version and #tostring(inline_version) <= MAX_LINE
nomsu = NomsuCode(tree.source) return inline_version
for i, bit in ipairs tree make_text = (tree)->
if type(bit) == 'string' nomsu = NomsuCode(tree.source)
-- TODO: unescape better? for i, bit in ipairs tree
bit = (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"')) if type(bit) == 'string'
bit = gsub(bit, "%G", ((c)-> c == ' ' and c or "\\#{c\byte!}")) bit = Parser.escape(bit)
nomsu\append bit bit_lines = files.get_lines(bit)
elseif bit.type == "Text" for j, line in ipairs bit_lines
nomsu\append(make_text(bit)) if j > 1 then nomsu\append "\n"
else line = gsub(line, "\\", "\\\\")
interp_nomsu = assert recurse(bit, inline:true) if #line > 1.25*MAX_LINE
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" remainder = line
interp_nomsu\parenthesize! while #remainder > 0
elseif bit.type == "Var" and type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]") split = find(remainder, " ", MAX_LINE, true)
interp_nomsu\parenthesize! if split
nomsu\append "\\", interp_nomsu chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1)
return nomsu nomsu\append chunk
return NomsuCode(tree.source, '"', make_text(tree), '"') elseif #remainder > 1.75*MAX_LINE
else split = math.floor(1.5*MAX_LINE)
inline_version = recurse(tree, inline:true) chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1)
if inline_version and #tostring(inline_version) <= MAX_LINE nomsu\append chunk
return inline_version else
make_text = (tree)-> nomsu\append remainder
nomsu = NomsuCode(tree.source) break
for i, bit in ipairs tree if #remainder > 0 then nomsu\append "\\\n.."
if type(bit) == 'string'
bit_lines = files.get_lines(bit)
for j, line in ipairs bit_lines
if j > 1 then nomsu\append "\n"
line = gsub(line, "\\", "\\\\")
if #line > 1.25*MAX_LINE
remainder = line
while #remainder > 0
split = find(remainder, " ", MAX_LINE, true)
if split
chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1)
nomsu\append chunk
elseif #remainder > 1.75*MAX_LINE
split = math.floor(1.5*MAX_LINE)
chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1)
nomsu\append chunk
else
nomsu\append remainder
break
if #remainder > 0 then nomsu\append "\\\n.."
else
nomsu\append line
elseif bit.type == "Text"
nomsu\append make_text(bit)
else
interp_nomsu = recurse(bit, inline:true)
if interp_nomsu
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict"
interp_nomsu\parenthesize!
elseif bit.type == "Var" and type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]")
interp_nomsu\parenthesize!
nomsu\append "\\", interp_nomsu
else else
interp_nomsu = assert(recurse(bit)) nomsu\append line
return nil unless interp_nomsu elseif bit.type == "Text"
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" and bit.type != "Block" nomsu\append make_text(bit)
nomsu\append "\\\n(..) ", interp_nomsu else
else interp_nomsu = @tree_to_inline_nomsu(bit)
if nomsu\trailing_line_len! + #tostring(interp_nomsu) <= MAX_LINE
switch bit.type
when "Var"
if type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]")
interp_nomsu\parenthesize!
when "List", "Dict"
nomsu\append "\\", interp_nomsu nomsu\append "\\", interp_nomsu
if i < #tree else
nomsu\append "\n.." nomsu\append "\\(", interp_nomsu, ")"
return nomsu else
return NomsuCode(tree.source, '".."\n ', make_text(tree)) interp_nomsu = recurse(bit)
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" and bit.type != "Block"
nomsu\append "\\(..)\n ", interp_nomsu
else
nomsu\append "\\", interp_nomsu
if i < #tree
nomsu\append "\n.."
return nomsu
return NomsuCode(tree.source, '".."\n ', make_text(tree))
when "List" when "List"
if options.inline inline_version = @tree_to_inline_nomsu tree
nomsu = NomsuCode(tree.source, "[") if inline_version and #tostring(inline_version) <= MAX_LINE
for i, item in ipairs tree return inline_version
nomsu\append ", " if i > 1 assert #tree > 0
nomsu\append assert(recurse(item, inline:true)) nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start))
nomsu\append "]" for i, item in ipairs tree
return nomsu item_nomsu = @tree_to_inline_nomsu(item)
else item_nomsu\parenthesize! if item.type == "Block"
inline_version = recurse(tree, inline:true) if nomsu\trailing_line_len! + #tostring(item_nomsu) <= MAX_LINE
if inline_version and #tostring(inline_version) <= MAX_LINE nomsu\append ", " if nomsu\trailing_line_len! > 0
return inline_version nomsu\append item_nomsu
assert #tree > 0 else
nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start)) item_nomsu = recurse(item)
for i, item in ipairs tree switch item.type
item_nomsu = assert recurse(item, inline:true) when "List", "Dict", "Text", "Block", "EscapedNomsu"
item_nomsu\parenthesize! if item.type == "Block" nomsu\append item_nomsu
if nomsu\trailing_line_len! + #tostring(item_nomsu) <= MAX_LINE
nomsu\append ", " if nomsu\trailing_line_len! > 0
nomsu\append item_nomsu
else
if #tostring(item_nomsu) > MAX_LINE
item_nomsu = recurse(item)
switch item.type
when "List", "Dict", "Text", "Block"
nomsu\append item_nomsu
else
nomsu\append "(..)\n ", item_nomsu
nomsu\append "\n" if i < #tree
else else
nomsu\append '\n' if nomsu\trailing_line_len! > 0 nomsu\append "(..)\n ", item_nomsu
nomsu\append pop_comments(item.source.start), item_nomsu nomsu\append "\n" if i < #tree
nomsu\append pop_comments(tree.source.stop, '\n') nomsu\append pop_comments(tree.source.stop, '\n')
return NomsuCode(tree.source, "[..]\n ", nomsu) return NomsuCode(tree.source, "[..]\n ", nomsu)
when "Dict" when "Dict"
if options.inline inline_version = @tree_to_inline_nomsu tree
nomsu = NomsuCode(tree.source, "{") if inline_version and #tostring(inline_version) <= MAX_LINE
for i, entry in ipairs tree return inline_version
nomsu\append ", " if i > 1 assert #tree > 0
nomsu\append assert(recurse(entry, inline:true)) nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start))
nomsu\append "}" for i, item in ipairs tree
return nomsu item_nomsu = @tree_to_inline_nomsu(item)
else item_nomsu\parenthesize! if item.type == "Block"
inline_version = recurse(tree, inline:true) if nomsu\trailing_line_len! + #tostring(item_nomsu) <= MAX_LINE
if inline_version and #tostring(inline_version) <= MAX_LINE nomsu\append ", " if nomsu\trailing_line_len! > 0
return inline_version nomsu\append item_nomsu
assert #tree > 0 else
nomsu = NomsuCode(tree.source, pop_comments(tree[1].source.start)) item_nomsu = recurse(item)
for i, item in ipairs tree switch item.type
item_nomsu = assert recurse(item, inline:true) when "List", "Dict", "Text", "Block", "EscapedNomsu"
item_nomsu\parenthesize! if item.type == "Block" nomsu\append item_nomsu
if nomsu\trailing_line_len! + #tostring(item_nomsu) <= MAX_LINE
nomsu\append ", " if nomsu\trailing_line_len! > 0
nomsu\append item_nomsu
else
if #tostring(item_nomsu) > MAX_LINE
item_nomsu = recurse(item)
switch item.type
when "List", "Dict", "Text", "Block"
nomsu\append item_nomsu
else
nomsu\append "(..)\n ", item_nomsu
nomsu\append "\n" if i < #tree
else else
nomsu\append '\n' if nomsu\trailing_line_len! > 0 nomsu\append "(..)\n ", item_nomsu
nomsu\append pop_comments(item.source.start), item_nomsu nomsu\append "\n" if i < #tree
nomsu\append pop_comments(tree.source.stop, '\n') nomsu\append pop_comments(tree.source.stop, '\n')
return NomsuCode(tree.source, "{..}\n ", nomsu) return NomsuCode(tree.source, "{..}\n ", nomsu)
when "DictEntry" when "DictEntry"
inline_version = @tree_to_inline_nomsu tree
if #tostring(inline_version) <= MAX_LINE
return inline_version
key, value = tree[1], tree[2] key, value = tree[1], tree[2]
key_nomsu = if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1]) key_nomsu = if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1])
NomsuCode(key.source, key[1]) NomsuCode(key.source, key[1])
else assert recurse(key, inline:true) else @tree_to_inline_nomsu(key)
key_nomsu\parenthesize! if key.type == "Action" or key.type == "Block" key_nomsu\parenthesize! if key.type == "Action" or key.type == "Block"
value_nomsu = if value
assert recurse(value, inline:true)
else NomsuCode(tree.source, "")
assert(value.type != "Block", "Didn't expect to find a Block as a value in a dict")
value_nomsu\parenthesize! if value.type == "Block"
if options.inline or #tostring(key_nomsu) + 2 + #tostring(value_nomsu) <= MAX_LINE
return NomsuCode tree.source, key_nomsu, ": ", value_nomsu
value_nomsu = recurse(value) value_nomsu = recurse(value)
if value.type == "List" or value.type == "Dict" or value.type == "Text" if value.type == "List" or value.type == "Dict" or value.type == "Text" or value.type == "Block"
return NomsuCode tree.source, key_nomsu, ": ", value_nomsu return NomsuCode tree.source, key_nomsu, ": ", value_nomsu
else else
return NomsuCode tree.source, key_nomsu, ": (..)\n ", value_nomsu return NomsuCode tree.source, key_nomsu, ": (..)\n ", value_nomsu
when "IndexChain" when "IndexChain", "Number", "Var"
nomsu = NomsuCode(tree.source) return @tree_to_inline_nomsu tree
for i, bit in ipairs tree
nomsu\append "." if i > 1
local bit_nomsu
bit_nomsu = if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' and bit[1]\match("[_a-zA-Z][_a-zA-Z0-9]*")
-- TODO: support arbitrary words here, including operators and unicode
bit[1]
else assert recurse(bit, inline:true)
switch bit.type
when "Action", "Block", "IndexChain"
bit_nomsu\parenthesize!
when "Number"
if i < #tree
bit_nomsu\parenthesize!
nomsu\append bit_nomsu
return nomsu
when "Number"
return NomsuCode(tree.source, tostring(tree[1]))
when "Var"
return NomsuCode(tree.source, "%", tree[1])
else else
error("Unknown type: #{tree.type}") error("Unknown type: #{tree.type}")

View File

@ -234,4 +234,22 @@ end
Parser.is_identifier = function(s) Parser.is_identifier = function(s)
return not not (NOMSU_DEFS.ident_char ^ 1 * -1):match(s) return not not (NOMSU_DEFS.ident_char ^ 1 * -1):match(s)
end end
local inline_escaper = re.compile("{~ (%utf8_char / ('\\' -> '\\\\') / [ -~] / ('\n' -> '\\n') / ('\t' -> '\\t') / ('\b' -> '\\b') / ('\a' -> '\\a') / ('\v' -> '\\v') / ('\f' -> '\\f') / ('\r' -> '\\r') / ('\"' -> '\\\"') / (. -> escape))* ~}", {
utf8_char = NOMSU_DEFS.utf8_char,
escape = (function(self)
return ("\\%03d"):format(self:byte())
end)
})
Parser.inline_escape = function(s)
return inline_escaper:match(s)
end
local escaper = re.compile("{~ (%utf8_char / ('\\' -> '\\\\') / [\n\r\t -~] / (. -> escape))* ~}", {
utf8_char = NOMSU_DEFS.utf8_char,
escape = (function(self)
return ("\\%03d"):format(self:byte())
end)
})
Parser.escape = function(s)
return escaper:match(s)
end
return Parser return Parser

View File

@ -144,4 +144,11 @@ Parser.is_operator = (s)->
Parser.is_identifier = (s)-> Parser.is_identifier = (s)->
return not not (NOMSU_DEFS.ident_char^1 * -1)\match(s) return not not (NOMSU_DEFS.ident_char^1 * -1)\match(s)
inline_escaper = re.compile "{~ (%utf8_char / ('\\' -> '\\\\') / [ -~] / ('\n' -> '\\n') / ('\t' -> '\\t') / ('\b' -> '\\b') / ('\a' -> '\\a') / ('\v' -> '\\v') / ('\f' -> '\\f') / ('\r' -> '\\r') / ('\"' -> '\\\"') / (. -> escape))* ~}", {utf8_char: NOMSU_DEFS.utf8_char, escape:(=> ("\\%03d")\format(@byte!))}
Parser.inline_escape = (s)->
return inline_escaper\match(s)
escaper = re.compile "{~ (%utf8_char / ('\\' -> '\\\\') / [\n\r\t -~] / (. -> escape))* ~}", {utf8_char: NOMSU_DEFS.utf8_char, escape:(=> ("\\%03d")\format(@byte!))}
Parser.escape = (s)->
return escaper\match(s)
return Parser return Parser

View File

@ -1,6 +1,17 @@
#!/usr/bin/env nomsu -V2.3.4.3 #!/usr/bin/env nomsu -V2.4.4.3
use "core" use "core"
use "lib/os.nom" use "lib/os.nom"
for %path in (=lua "arg"): %args = (command line args)
%inplace = (no)
if (%args.1 is "-i"):
%inplace = (yes)
remove index 1 from %args
for %path in %args:
for file %filename in %path: for file %filename in %path:
say ((parse (read file %filename) from %filename) as nomsu) %formatted = ".."
#!/usr/bin/env nomsu -V\(Nomsu version)
\((parse (read file %filename) from %filename) as nomsu)
if %inplace: write %formatted to file %filename
..else: say %formatted