426 lines
14 KiB
Lua
426 lines
14 KiB
Lua
local NomsuCode
|
|
NomsuCode = require("code_obj").NomsuCode
|
|
local find, sub, match
|
|
do
|
|
local _obj_0 = string
|
|
find, sub, match = _obj_0.find, _obj_0.sub, _obj_0.match
|
|
end
|
|
local R, P, S
|
|
do
|
|
local _obj_0 = require('lpeg')
|
|
R, P, S = _obj_0.R, _obj_0.P, _obj_0.S
|
|
end
|
|
local re = require('re')
|
|
local MAX_LINE = 90
|
|
local utf8_char_patt = (R("\194\223") * R("\128\191") + R("\224\239") * R("\128\191") * R("\128\191") + R("\240\244") * R("\128\191") * R("\128\191") * R("\128\191"))
|
|
local operator_patt = S("'`~!@$^&*+=|<>?/-") ^ 1 * -1
|
|
local identifier_patt = (R("az", "AZ", "09") + P("_") + utf8_char_patt) ^ 1 * -1
|
|
local is_operator
|
|
is_operator = function(s)
|
|
return not not operator_patt:match(s)
|
|
end
|
|
local is_identifier
|
|
is_identifier = function(s)
|
|
return not not identifier_patt:match(s)
|
|
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 = utf8_char_patt,
|
|
escape = (function(self)
|
|
return ("\\%03d"):format(self:byte())
|
|
end)
|
|
})
|
|
local inline_escape
|
|
inline_escape = function(s)
|
|
return inline_escaper:match(s)
|
|
end
|
|
local escaper = re.compile("{~ (%utf8_char / ('\\' -> '\\\\') / [\n\r\t -~] / (. -> escape))* ~}", {
|
|
utf8_char = utf8_char_patt,
|
|
escape = (function(self)
|
|
return ("\\%03d"):format(self:byte())
|
|
end)
|
|
})
|
|
local escape
|
|
escape = function(s)
|
|
return escaper:match(s)
|
|
end
|
|
local tree_to_inline_nomsu
|
|
tree_to_inline_nomsu = function(tree)
|
|
local _exp_0 = tree.type
|
|
if "Action" == _exp_0 then
|
|
local nomsu = NomsuCode:from(tree.source)
|
|
if tree.target then
|
|
local inline_target = tree_to_inline_nomsu(tree.target)
|
|
if tree.target.type == "Action" then
|
|
inline_target:parenthesize()
|
|
end
|
|
nomsu:append(inline_target, "::")
|
|
end
|
|
for i, bit in ipairs(tree) do
|
|
if type(bit) == "string" then
|
|
local clump_words
|
|
if type(tree[i - 1]) then
|
|
clump_words = is_operator(bit) ~= is_operator(tree[i - 1])
|
|
else
|
|
clump_words = is_operator(bit)
|
|
end
|
|
if i > 1 and not clump_words then
|
|
nomsu:append(" ")
|
|
end
|
|
nomsu:append(bit)
|
|
else
|
|
local arg_nomsu = tree_to_inline_nomsu(bit)
|
|
if bit.type == "Block" then
|
|
if i > 1 and i < #tree then
|
|
nomsu:append(" ")
|
|
end
|
|
if not (i == #tree) then
|
|
arg_nomsu:parenthesize()
|
|
end
|
|
else
|
|
if i > 1 then
|
|
nomsu:append(" ")
|
|
end
|
|
if bit.type == "Action" then
|
|
arg_nomsu:parenthesize()
|
|
end
|
|
end
|
|
nomsu:append(arg_nomsu)
|
|
end
|
|
end
|
|
return nomsu
|
|
elseif "EscapedNomsu" == _exp_0 then
|
|
local inner_nomsu = tree_to_inline_nomsu(tree[1])
|
|
if not (tree[1].type == "List" or tree[1].type == "Dict" or tree[1].type == "Var") then
|
|
inner_nomsu:parenthesize()
|
|
end
|
|
return NomsuCode:from(tree.source, "\\", inner_nomsu)
|
|
elseif "Block" == _exp_0 then
|
|
local nomsu = NomsuCode:from(tree.source, ":")
|
|
for i, line in ipairs(tree) do
|
|
nomsu:append(i == 1 and " " or "; ")
|
|
nomsu:append(tree_to_inline_nomsu(line))
|
|
end
|
|
if #tree > 1 then
|
|
nomsu:parenthesize()
|
|
end
|
|
return nomsu
|
|
elseif "Text" == _exp_0 then
|
|
local add_text
|
|
add_text = function(nomsu, tree)
|
|
for i, bit in ipairs(tree) do
|
|
if type(bit) == 'string' then
|
|
local escaped = inline_escape(bit)
|
|
nomsu:append(inline_escape(bit))
|
|
elseif bit.type == "Text" then
|
|
add_text(nomsu, bit)
|
|
else
|
|
local interp_nomsu = 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
|
|
end
|
|
local nomsu = NomsuCode:from(tree.source)
|
|
add_text(nomsu, tree)
|
|
return NomsuCode:from(tree.source, '"', nomsu, '"')
|
|
elseif "List" == _exp_0 or "Dict" == _exp_0 then
|
|
local nomsu = NomsuCode:from(tree.source, (tree.type == "List" and "[" or "{"))
|
|
for i, item in ipairs(tree) do
|
|
if i > 1 then
|
|
nomsu:append(", ")
|
|
end
|
|
nomsu:append(tree_to_inline_nomsu(item))
|
|
end
|
|
nomsu:append(tree.type == "List" and "]" or "}")
|
|
return nomsu
|
|
elseif "DictEntry" == _exp_0 then
|
|
local key, value = tree[1], tree[2]
|
|
local nomsu
|
|
if key.type == "Text" and #key == 1 and is_identifier(key[1]) then
|
|
nomsu = NomsuCode:from(key.source, key[1])
|
|
else
|
|
nomsu = tree_to_inline_nomsu(key)
|
|
end
|
|
if key.type == "Action" or key.type == "Block" then
|
|
nomsu:parenthesize()
|
|
end
|
|
assert(value.type ~= "Block", "Didn't expect to find a Block as a value in a dict")
|
|
nomsu:append(":")
|
|
if value then
|
|
local value_nomsu = tree_to_inline_nomsu(value)
|
|
if value.type == "Block" then
|
|
value_nomsu:parenthesize()
|
|
end
|
|
nomsu:append(value_nomsu)
|
|
end
|
|
return nomsu
|
|
elseif "IndexChain" == _exp_0 then
|
|
local nomsu = NomsuCode:from(tree.source)
|
|
for i, bit in ipairs(tree) do
|
|
if i > 1 then
|
|
nomsu:append(".")
|
|
end
|
|
local bit_nomsu
|
|
if i > 1 and bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' and is_identifier(bit[1]) then
|
|
bit_nomsu = bit[1]
|
|
else
|
|
bit_nomsu = tree_to_inline_nomsu(bit)
|
|
end
|
|
assert(bit.type ~= "Block")
|
|
if bit.type == "Action" or bit.type == "IndexChain" or (bit.type == "Number" and i < #tree) then
|
|
bit_nomsu:parenthesize()
|
|
end
|
|
nomsu:append(bit_nomsu)
|
|
end
|
|
return nomsu
|
|
elseif "Number" == _exp_0 then
|
|
return NomsuCode:from(tree.source, tostring(tree[1]))
|
|
elseif "Var" == _exp_0 then
|
|
return NomsuCode:from(tree.source, "%", tree[1])
|
|
elseif "FileChunks" == _exp_0 then
|
|
return error("Can't inline a FileChunks")
|
|
elseif "Comment" == _exp_0 then
|
|
return NomsuCode:from(tree.source)
|
|
elseif "Error" == _exp_0 then
|
|
return error("Can't compile errors")
|
|
else
|
|
return error("Unknown type: " .. tostring(tree.type))
|
|
end
|
|
end
|
|
local tree_to_nomsu
|
|
tree_to_nomsu = function(tree)
|
|
local nomsu = NomsuCode:from(tree.source)
|
|
local recurse
|
|
recurse = function(t)
|
|
local space = MAX_LINE - nomsu:trailing_line_len()
|
|
local inline = true
|
|
for subtree in coroutine.wrap(function()
|
|
return (t:map(coroutine.yield) and nil)
|
|
end) do
|
|
if subtree.type == "Block" then
|
|
if #subtree > 1 or #tree_to_inline_nomsu(subtree):text() > 20 then
|
|
inline = false
|
|
end
|
|
end
|
|
end
|
|
if inline then
|
|
local inline_nomsu = tree_to_inline_nomsu(t)
|
|
if #inline_nomsu:text() <= space then
|
|
if t.type == "Action" then
|
|
inline_nomsu:parenthesize()
|
|
end
|
|
return inline_nomsu
|
|
end
|
|
end
|
|
local indented = tree_to_nomsu(t)
|
|
if t.type == "Action" then
|
|
if indented:is_multiline() then
|
|
return NomsuCode:from(t.source, "(..)\n ", indented)
|
|
else
|
|
indented:parenthesize()
|
|
end
|
|
end
|
|
return indented
|
|
end
|
|
local _exp_0 = tree.type
|
|
if "FileChunks" == _exp_0 then
|
|
if tree.shebang then
|
|
nomsu:append(tree.shebang, "\n")
|
|
end
|
|
local should_clump
|
|
should_clump = function(prev_line, line)
|
|
if prev_line.type == "Action" and line.type == "Action" then
|
|
if prev_line.stub == "use" then
|
|
return line.stub == "use"
|
|
end
|
|
if prev_line.stub == "test" then
|
|
return true
|
|
end
|
|
if line.stub == "test" then
|
|
return false
|
|
end
|
|
end
|
|
return not tree_to_nomsu(prev_line):is_multiline()
|
|
end
|
|
for chunk_no, chunk in ipairs(tree) do
|
|
if chunk_no > 1 then
|
|
nomsu:append("\n\n" .. tostring(("~"):rep(80)) .. "\n\n")
|
|
end
|
|
if chunk.type == "Block" then
|
|
for line_no, line in ipairs(chunk) do
|
|
if line_no > 1 then
|
|
if should_clump(chunk[line_no - 1], line) then
|
|
nomsu:append("\n")
|
|
else
|
|
nomsu:append("\n\n")
|
|
end
|
|
end
|
|
nomsu:append(tree_to_nomsu(line))
|
|
end
|
|
else
|
|
nomsu:append(tree_to_nomsu(chunk))
|
|
end
|
|
end
|
|
if not (nomsu:match("\n$")) then
|
|
nomsu:append('\n')
|
|
end
|
|
return nomsu
|
|
elseif "Action" == _exp_0 then
|
|
local next_space = ""
|
|
if tree.target then
|
|
local target_nomsu = recurse(tree.target)
|
|
if tree.target.type == "Block" and not target_nomsu:is_multiline() then
|
|
target_nomsu:parenthesize()
|
|
end
|
|
nomsu:append(target_nomsu)
|
|
nomsu:append(target_nomsu:is_multiline() and "\n..::" or "::")
|
|
end
|
|
for i, bit in ipairs(tree) do
|
|
if type(bit) == "string" then
|
|
if not (next_space == " " and (type(tree[i - 1]) == 'string' and is_operator(tree[i - 1]) ~= is_operator(bit))) then
|
|
nomsu:append(next_space)
|
|
end
|
|
nomsu:append(bit)
|
|
next_space = nomsu:trailing_line_len() > MAX_LINE and " \\\n.." or " "
|
|
else
|
|
local bit_nomsu = recurse(bit)
|
|
if i < #tree and bit.type == "Block" and not bit_nomsu:is_multiline() then
|
|
bit_nomsu:parenthesize()
|
|
end
|
|
if next_space == " " and not bit_nomsu:is_multiline() and nomsu:trailing_line_len() + #bit_nomsu:text() > MAX_LINE then
|
|
if bit.type == 'Action' then
|
|
bit_nomsu = NomsuCode:from(bit.source, "(..)\n ", tree_to_nomsu(bit))
|
|
else
|
|
next_space = " \\\n.."
|
|
end
|
|
end
|
|
if not (next_space == " " and bit.type == "Block") then
|
|
nomsu:append(next_space)
|
|
end
|
|
nomsu:append(bit_nomsu)
|
|
next_space = bit_nomsu:is_multiline() and "\n.." or " "
|
|
end
|
|
end
|
|
return nomsu
|
|
elseif "EscapedNomsu" == _exp_0 then
|
|
nomsu = recurse(tree[1])
|
|
if tree[1].type == 'Block' and not nomsu:is_multiline() then
|
|
nomsu:parenthesize()
|
|
end
|
|
return NomsuCode:from(tree.source, "\\", nomsu)
|
|
elseif "Block" == _exp_0 then
|
|
for i, line in ipairs(tree) do
|
|
local line_nomsu = tree_to_nomsu(line)
|
|
nomsu:append(line_nomsu)
|
|
if i < #tree then
|
|
nomsu:append(line_nomsu:match('\n[^\n]*\n') and "\n\n" or "\n")
|
|
end
|
|
end
|
|
return NomsuCode:from(tree.source, ":\n ", nomsu)
|
|
elseif "Text" == _exp_0 then
|
|
local max_line = math.floor(1.25 * MAX_LINE)
|
|
local add_text
|
|
add_text = function(tree)
|
|
for i, bit in ipairs(tree) do
|
|
if type(bit) == 'string' then
|
|
bit = escape(bit)
|
|
for j, line in ipairs(bit:lines()) do
|
|
if j > 1 then
|
|
nomsu:append("\n")
|
|
elseif #line > 10 and nomsu:trailing_line_len() > max_line then
|
|
nomsu:append("\\\n..")
|
|
end
|
|
while #line > 0 do
|
|
local space = max_line - nomsu:trailing_line_len()
|
|
local split = find(line, "[%p%s]", space)
|
|
if not split or split > space + 10 then
|
|
split = space + 10
|
|
end
|
|
if #line - split < 10 then
|
|
split = #line
|
|
end
|
|
local bite
|
|
bite, line = sub(line, 1, split), sub(line, split + 1, -1)
|
|
nomsu:append(bite)
|
|
if #line > 0 then
|
|
nomsu:append("\\\n..")
|
|
end
|
|
end
|
|
end
|
|
elseif bit.type == "Text" then
|
|
add_text(bit)
|
|
else
|
|
nomsu:append("\\")
|
|
local interp_nomsu = recurse(bit)
|
|
if not (interp_nomsu:is_multiline()) then
|
|
if bit.type == "Var" then
|
|
if type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
|
|
interp_nomsu:parenthesize()
|
|
end
|
|
elseif bit.type == "EscapedNomsu" or bit.type == "Block" then
|
|
interp_nomsu:parenthesize()
|
|
end
|
|
end
|
|
nomsu:append(interp_nomsu)
|
|
if interp_nomsu:is_multiline() then
|
|
nomsu:append("\n..")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
add_text(tree)
|
|
return NomsuCode:from(tree.source, '"\\\n ..', nomsu, '"')
|
|
elseif "List" == _exp_0 or "Dict" == _exp_0 then
|
|
if #tree == 0 then
|
|
nomsu:append(tree.type == "List" and "[]" or "{}")
|
|
return nomsu
|
|
end
|
|
for i, item in ipairs(tree) do
|
|
local item_nomsu = tree_to_inline_nomsu(item)
|
|
if #item_nomsu:text() > MAX_LINE then
|
|
item_nomsu = recurse(item)
|
|
end
|
|
nomsu:append(item_nomsu)
|
|
if i < #tree then
|
|
nomsu:append((item_nomsu:is_multiline() or nomsu:trailing_line_len() + #tostring(item_nomsu) >= MAX_LINE) and '\n' or ', ')
|
|
end
|
|
end
|
|
if tree.type == "List" then
|
|
return NomsuCode:from(tree.source, "[..]\n ", nomsu)
|
|
else
|
|
return NomsuCode:from(tree.source, "{..}\n ", nomsu)
|
|
end
|
|
elseif "DictEntry" == _exp_0 then
|
|
local key, value = tree[1], tree[2]
|
|
if key.type == "Text" and #key == 1 and is_identifier(key[1]) then
|
|
nomsu = NomsuCode:from(key.source, key[1])
|
|
else
|
|
nomsu = tree_to_inline_nomsu(key)
|
|
end
|
|
if key.type == "Block" then
|
|
nomsu:parenthesize()
|
|
end
|
|
local value_nomsu = tree_to_nomsu(value)
|
|
if (value.type == "Block" or value.type == "EscapedNomsu") and not value_nomsu:is_multiline() then
|
|
value_nomsu:parenthesize()
|
|
end
|
|
nomsu:append(": ", value_nomsu)
|
|
return nomsu
|
|
elseif "Comment" == _exp_0 then
|
|
nomsu:append("#", (tree[1]:gsub("\n", "\n ")))
|
|
return nomsu
|
|
elseif "IndexChain" == _exp_0 or "Number" == _exp_0 or "Var" == _exp_0 or "Comment" == _exp_0 or "Error" == _exp_0 then
|
|
return tree_to_inline_nomsu(tree)
|
|
else
|
|
return error("Unknown type: " .. tostring(tree.type))
|
|
end
|
|
end
|
|
return {
|
|
tree_to_nomsu = tree_to_nomsu,
|
|
tree_to_inline_nomsu = tree_to_inline_nomsu
|
|
}
|