Fixed nomsu codegen for indented text and improved text parsing. Also

moved some more tests inline.
This commit is contained in:
Bruce Hill 2018-07-11 14:13:43 -07:00
parent b32634faf8
commit ba2b83d566
7 changed files with 154 additions and 117 deletions

View File

@ -182,6 +182,15 @@ compile [%dict with fallback %key -> %value] to
end}) end})
# Sorting # Sorting
test
%x <- [3,1,2]
sort %x
assume: %x = [1,2,3]
sort %x by % = (-%)
assume: %x = [3,2,1]
%keys <- {1:999,2:0,3:50}
sort %x by % = %keys.%
assume: %x = [2,3,1]
compile [sort %items] to: Lua "table.sort(\(%items as lua expr));" compile [sort %items] to: Lua "table.sort(\(%items as lua expr));"
parse [..] parse [..]
sort %items by %item = %key_expr sort %items by %item = %key_expr
@ -193,6 +202,7 @@ parse [..]
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
test: assume: (sorted [3,1,2]) = [1,2,3]
action [%items sorted, sorted %items] action [%items sorted, sorted %items]
%copy <- (% for % in %items) %copy <- (% for % in %items)
sort %copy sort %copy
@ -206,6 +216,7 @@ parse [..]
sort %copy by %item = %key sort %copy by %item = %key
return %copy return %copy
test: assume: (unique [1,2,1,3,2,3]) = [1,2,3]
action [unique %items] action [unique %items]
%unique <- [] %unique <- []
%seen <- {} %seen <- {}

View File

@ -8,14 +8,17 @@ use "core/operators.nom"
use "core/errors.nom" use "core/errors.nom"
# No-Op # No-Op
test: do nothing
compile [do nothing] to: Lua "" compile [do nothing] to: Lua ""
# Conditionals # Conditionals
test: if (no): barf "conditional fail"
compile [if %condition %if_body] to compile [if %condition %if_body] to
Lua ".." Lua ".."
if \(%condition as lua expr) then if \(%condition as lua expr) then
\(%if_body as lua statements) \(%if_body as lua statements)
end end
test: unless (yes): barf "conditional fail"
parse [unless %condition %unless_body] as: if (not %condition) %unless_body parse [unless %condition %unless_body] as: if (not %condition) %unless_body
compile [if %condition %if_body else %else_body, unless %condition %else_body else %if_body] to compile [if %condition %if_body else %else_body, unless %condition %else_body else %if_body] to

View File

@ -44,6 +44,9 @@ compile [%expr for %match where %text matches %patt] to
return ret return ret
end)() end)()
compile [%text matches %pattern] to
Lua value "(\(%text as lua expr):match(\(%pattern as lua expr)) and true or false)"
# Text literals # Text literals
lua> ".." lua> ".."
do do

View File

@ -1,5 +1,6 @@
use "core" use "core"
test: bright: "\(green)Color test passed."
%colors <- {..} %colors <- {..}
normal:0, "reset color":0, bright:1, bold:1, dim:2, italic:3, underscore:4 normal:0, "reset color":0, bright:1, bold:1, dim:2, italic:3, underscore:4
"slow blink":5, "fast blink": 6, reverse:7, inverse:7, inverted:7, hidden:8 "slow blink":5, "fast blink": 6, reverse:7, inverse:7, inverted:7, hidden:8
@ -14,4 +15,4 @@ for %name = %colornum in %colors
compile [\%name] to: Lua value (quote \(quote %escapecode)) compile [\%name] to: Lua value (quote \(quote %escapecode))
compile [\%name %text] to compile [\%name %text] to
Lua value ".." Lua value ".."
(\\(quote \(quote %escapecode))..\\(%text as lua expr)..\(quote "\27[0m")) (\\(quote \(quote %escapecode))..\\(%text as lua expr).."\\27[0m")

View File

@ -101,11 +101,11 @@ inline_text_interpolation:
indented_text (Text): indented_text (Text):
'".."' eol %nl {| '".."' eol %nl {|
{~ (%nl*) (%indent -> "") ~} {~ (%nl*) (%indent -> "") ~}
(indented_plain_text / text_interpolation / {~ ("\" nodent "..") -> "" ~} / {~ %nl+ (%nodent -> "") ~})* (indented_plain_text / text_interpolation / {~ %nl+ (%nodent -> "") ~})*
|} (((!.) %dedent) / (&(%nl %dedent)) / (({} (non_dedent_error -> "Unexpected character while parsing Text") %userdata) => error)) |} (((!.) %dedent) / (&(%nl %dedent)) / (({} (non_dedent_error -> "Unexpected character while parsing Text") %userdata) => error))
indented_plain_text (Text): indented_plain_text (Text):
{| ({~ "\\" -> "\" ~} / {[^%nl\]+} / {!(text_interpolation / "\" nodent "..") "\"})+ {| {~ (("\\" -> "\") / (("\" nodent "..") -> "") / (!text_interpolation "\") / [^%nl\]+)+
{~ (%nl+ (%nodent -> "")) / (("\" nodent "..") -> "") ~}* |} (%nl+ (%nodent -> ""))* ~} |}
text_interpolation: text_interpolation:
inline_text_interpolation / ("\" indented_expression nodent "..") inline_text_interpolation / ("\" indented_expression nodent "..")

View File

@ -786,12 +786,16 @@ do
if not (arg_nomsu) then if not (arg_nomsu) then
return nil return nil
end end
if bit.type == "Action" or bit.type == "Block" then
if bit.type == "Action" and i == #tree then
nomsu:append(":")
else
arg_nomsu:parenthesize()
end
end
if not (i == 1) then if not (i == 1) then
nomsu:append(" ") nomsu:append(" ")
end end
if bit.type == "Action" or bit.type == "Block" then
arg_nomsu:parenthesize()
end
nomsu:append(arg_nomsu) nomsu:append(arg_nomsu)
end end
end end
@ -904,27 +908,28 @@ do
return nomsu return nomsu
elseif "Text" == _exp_0 then elseif "Text" == _exp_0 then
if inline then if inline then
local nomsu = NomsuCode(tree.source, '"') local make_text
make_text = function(tree)
local nomsu = NomsuCode(tree.source)
for _index_0 = 1, #tree do for _index_0 = 1, #tree do
local bit = tree[_index_0] local bit = tree[_index_0]
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"), '"', '\\"')))
elseif bit.type == "Text" then
nomsu:append(make_text(bit))
else else
local interp_nomsu = recurse(bit, { local interp_nomsu = assert(recurse(bit, {
inline = true inline = true
}) }))
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
return nil
end end
end end
end
nomsu:append('"')
return nomsu return nomsu
end
return NomsuCode(tree.source, '"', make_text(tree), '"')
else else
local inline_version = recurse(tree, { local inline_version = recurse(tree, {
inline = true inline = true
@ -932,7 +937,9 @@ do
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, '".."\n ') local make_text
make_text = function(tree)
local nomsu = NomsuCode(tree.source)
for i, bit in ipairs(tree) do for i, bit in ipairs(tree) do
if type(bit) == 'string' then if type(bit) == 'string' then
local bit_lines = files.get_lines(bit) local bit_lines = files.get_lines(bit)
@ -965,6 +972,8 @@ do
nomsu:append(line) nomsu:append(line)
end end
end end
elseif bit.type == "Text" then
nomsu:append(make_text(bit))
else else
local interp_nomsu = recurse(bit, { local interp_nomsu = recurse(bit, {
inline = true inline = true
@ -988,6 +997,8 @@ do
end end
return nomsu return nomsu
end end
return NomsuCode(tree.source, '".."\n ', make_text(tree))
end
elseif "List" == _exp_0 then elseif "List" == _exp_0 then
if inline then if inline then
local nomsu = NomsuCode(tree.source, "[") local nomsu = NomsuCode(tree.source, "[")

View File

@ -522,10 +522,13 @@ with NomsuCompiler
else else
arg_nomsu = recurse(bit,inline:true) arg_nomsu = recurse(bit,inline:true)
return nil unless arg_nomsu return nil unless arg_nomsu
if bit.type == "Action" or bit.type == "Block"
if bit.type == "Action" and i == #tree
nomsu\append ":"
else
arg_nomsu\parenthesize!
unless i == 1 unless i == 1
nomsu\append " " nomsu\append " "
if bit.type == "Action" or bit.type == "Block"
arg_nomsu\parenthesize!
nomsu\append arg_nomsu nomsu\append arg_nomsu
return nomsu return nomsu
else else
@ -607,25 +610,27 @@ with NomsuCompiler
when "Text" when "Text"
if inline if inline
nomsu = NomsuCode(tree.source, '"') make_text = (tree)->
nomsu = NomsuCode(tree.source)
for bit in *tree for bit in *tree
if type(bit) == 'string' if type(bit) == 'string'
-- TODO: unescape better? -- TODO: unescape better?
nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"')) nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"'))
elseif bit.type == "Text"
nomsu\append(make_text(bit))
else else
interp_nomsu = recurse(bit, inline:true) interp_nomsu = assert recurse(bit, inline:true)
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 return nil
nomsu\append '"'
return nomsu return nomsu
return NomsuCode(tree.source, '"', make_text(tree), '"')
else else
inline_version = recurse(tree, inline: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 ') make_text = (tree)->
nomsu = NomsuCode(tree.source)
for i, bit in ipairs tree for i, bit in ipairs tree
if type(bit) == 'string' if type(bit) == 'string'
bit_lines = files.get_lines(bit) bit_lines = files.get_lines(bit)
@ -648,6 +653,8 @@ with NomsuCompiler
if #remainder > 0 then nomsu\append "\\\n.." if #remainder > 0 then nomsu\append "\\\n.."
else else
nomsu\append line nomsu\append line
elseif bit.type == "Text"
nomsu\append make_text(bit)
else else
interp_nomsu = recurse(bit, inline:true) interp_nomsu = recurse(bit, inline:true)
if interp_nomsu if interp_nomsu
@ -661,6 +668,7 @@ with NomsuCompiler
if i < #tree if i < #tree
nomsu\append "\n.." nomsu\append "\n.."
return nomsu return nomsu
return NomsuCode(tree.source, '".."\n ', make_text(tree))
when "List" when "List"
if inline if inline