Working string interpolation!
This commit is contained in:
parent
0f228d2d25
commit
83463f11c5
8
core.nom
8
core.nom
@ -115,13 +115,6 @@ rule "%a != %b":
|
|||||||
rule "say %str":
|
rule "say %str":
|
||||||
lua block ["print(utils.repr(", %str, "))"]
|
lua block ["print(utils.repr(", %str, "))"]
|
||||||
|
|
||||||
rule "printf %str":
|
|
||||||
lua block ".."
|
|
||||||
|for _,s in ipairs(vars.str) do
|
|
||||||
| io.write(utils.repr(s))
|
|
||||||
|end
|
|
||||||
|io.write("\n")
|
|
||||||
|
|
||||||
rule "do %action":
|
rule "do %action":
|
||||||
lua expr "vars.action(compiler, setmetatable({}, {__index=vars}))"
|
lua expr "vars.action(compiler, setmetatable({}, {__index=vars}))"
|
||||||
|
|
||||||
@ -161,6 +154,7 @@ macro block "for %varname in %iterable %body":
|
|||||||
"\n vars[", %varname, "] = old_loopval"
|
"\n vars[", %varname, "] = old_loopval"
|
||||||
"\nend"
|
"\nend"
|
||||||
|
|
||||||
|
|
||||||
rule "%start up to %stop":
|
rule "%start up to %stop":
|
||||||
lua expr "utils.range(vars.start,vars.stop-1)"
|
lua expr "utils.range(vars.start,vars.stop-1)"
|
||||||
|
|
||||||
|
@ -104,12 +104,11 @@ if %x:
|
|||||||
say "three"
|
say "three"
|
||||||
|
|
||||||
|
|
||||||
printf [..]
|
say ".."
|
||||||
".."
|
|this is a longstring
|
||||||
|this is a longstring
|
|
|
||||||
|
|
| with multiple lines
|
||||||
.., "with", ".."
|
| and an interpolated expression: \2 + 5\
|
||||||
| multiple lines
|
|
||||||
|
|
||||||
rule "%n bottles":
|
rule "%n bottles":
|
||||||
lua block [..]
|
lua block [..]
|
||||||
|
@ -89,7 +89,8 @@ with everyone's approval:
|
|||||||
..else:
|
..else:
|
||||||
let "approvers" = (number of (* %pending = "approved"))
|
let "approvers" = (number of (* %pending = "approved"))
|
||||||
let "num-players" = (number of (players))
|
let "num-players" = (number of (players))
|
||||||
printf [%approvers, "/", %num-players, " players have approved"]
|
say ".."
|
||||||
|
|\number of (* %pending = "approved")\/\number of (players)\ players have approved.
|
||||||
|
|
||||||
rule ["reject", "vote no", "vote nay", "veto", "disapprove"]:
|
rule ["reject", "vote no", "vote nay", "veto", "disapprove"]:
|
||||||
let "pending" = ("pending proposal" "is" ?)
|
let "pending" = ("pending proposal" "is" ?)
|
||||||
@ -104,7 +105,8 @@ with everyone's approval:
|
|||||||
|
|
||||||
rule "join":
|
rule "join":
|
||||||
(you) "is a player" = (yes)
|
(you) "is a player" = (yes)
|
||||||
printf ["Welcome to the game, ", you,"!"]
|
say ".."
|
||||||
|
|Welcome to the game, \you\!
|
||||||
|
|
||||||
allow "unpropose" to use "set all * % = %"
|
allow "unpropose" to use "set all * % = %"
|
||||||
allow ["join", "mark % as approving %", "mark % as rejecting %", "propose %", "unpropose"] to use "% % = %"
|
allow ["join", "mark % as approving %", "mark % as rejecting %", "propose %", "unpropose"] to use "% % = %"
|
||||||
@ -149,9 +151,11 @@ propose:
|
|||||||
rule "vote for %candidate":
|
rule "vote for %candidate":
|
||||||
(you) "votes for" = %candidate
|
(you) "votes for" = %candidate
|
||||||
let "vote-percent" = ((number of (* "votes for" = %candidate)) / (number of (players)))
|
let "vote-percent" = ((number of (* "votes for" = %candidate)) / (number of (players)))
|
||||||
printf ["Vote cast. ",%candidate," now has ",(100 * %vote-percent),"% of the votes."]
|
say ".."
|
||||||
|
|Vote cast. \%candidate\ now has \100 * %vote-percent\% of the votes.
|
||||||
if (%vote-percent > 0.5):
|
if (%vote-percent > 0.5):
|
||||||
printf ["The winner of the election is:", %candidate]
|
say ".."
|
||||||
|
|The winner of the election is: \%candidate\
|
||||||
close the election
|
close the election
|
||||||
|
|
||||||
allow ["open election %", "close the election", "vote for %"] to use ["% % = %"]
|
allow ["open election %", "close the election", "vote for %"] to use ["% % = %"]
|
||||||
@ -163,7 +167,8 @@ propose:
|
|||||||
if ((you) == "Anonymous"):
|
if ((you) == "Anonymous"):
|
||||||
make "bill" do %action
|
make "bill" do %action
|
||||||
..else:
|
..else:
|
||||||
printf ["Who do you think you are, ", you,"?"]
|
say ".."
|
||||||
|
|Who do you think you are, \you\?
|
||||||
allow ["as bill %"] to use ["make % %"]
|
allow ["as bill %"] to use ["make % %"]
|
||||||
approve
|
approve
|
||||||
as bill: join
|
as bill: join
|
||||||
@ -173,7 +178,8 @@ propose:
|
|||||||
if ((you) == "Anonymous"):
|
if ((you) == "Anonymous"):
|
||||||
make "dave" do %action
|
make "dave" do %action
|
||||||
..else:
|
..else:
|
||||||
printf ["Who do you think you are, ", you,"?"]
|
say ".."
|
||||||
|
|Who do you think you are, \you\?
|
||||||
allow ["as dave %"] to use ["make % %"]
|
allow ["as dave %"] to use ["make % %"]
|
||||||
approve
|
approve
|
||||||
as bill: approve
|
as bill: approve
|
||||||
|
@ -13,8 +13,7 @@ run file "core.nom"
|
|||||||
|
|
||||||
# Strings:
|
# Strings:
|
||||||
"asdf"
|
"asdf"
|
||||||
".."
|
".."|This is a multi-line string with a #.. fake comment
|
||||||
|This is a multi-line string with a #.. fake comment
|
|
||||||
|that starts with ".." and includes each indented line that starts with a "|"
|
|that starts with ".." and includes each indented line that starts with a "|"
|
||||||
|until the indentation ends
|
|until the indentation ends
|
||||||
|
|
||||||
@ -101,14 +100,22 @@ say both ".."
|
|||||||
"-- Abraham Lincoln"
|
"-- Abraham Lincoln"
|
||||||
|
|
||||||
rule "my favorite number": return 23
|
rule "my favorite number": return 23
|
||||||
|
|
||||||
# Subexpressions are wrapped in parentheses:
|
# Subexpressions are wrapped in parentheses:
|
||||||
# printf takes a list of bits that are converted to strings and concatenated together, and printed
|
say (my favorite number)
|
||||||
printf ["My favorite number is ", my favorite number]
|
|
||||||
# There's a multi-line indented block form for subexpressions too:
|
# There's a multi-line indented block form for subexpressions too:
|
||||||
printf [..]
|
say (..)
|
||||||
"My favorite number is still ", (..)
|
my favorite
|
||||||
|
number
|
||||||
|
|
||||||
|
# Block strings can interpolate values by enclosing an expression in a pair of \s
|
||||||
|
say ".."|My favorite number is \my favorite number\!
|
||||||
|
|
||||||
|
say ".."
|
||||||
|
|My favorite number is still \(..)
|
||||||
my favorite
|
my favorite
|
||||||
number
|
number
|
||||||
|
..\, but this time it uses an indented subexpression!
|
||||||
|
|
||||||
#.. There's a few macros in the language for things like conditional branches and logic/math
|
#.. There's a few macros in the language for things like conditional branches and logic/math
|
||||||
operations, but they can be thought of as basically the same as functions.
|
operations, but they can be thought of as basically the same as functions.
|
||||||
@ -146,17 +153,15 @@ if (1 > 10):
|
|||||||
let "numbers" = [5,6,7]
|
let "numbers" = [5,6,7]
|
||||||
|
|
||||||
# Looping:
|
# Looping:
|
||||||
printf ["Looping over: ",%numbers,"!"]
|
say ".."|Looping over \%numbers\:
|
||||||
for "number" in %numbers:
|
for "number" in %numbers:
|
||||||
say (%number + 100)
|
say (%number + 100)
|
||||||
|
|
||||||
rule "sing %starting-bottles bottles of beer":
|
rule "sing %starting-bottles bottles of beer":
|
||||||
for "n" in (%starting-bottles down through 0):
|
for "n" in (%starting-bottles down through 0):
|
||||||
printf [..]
|
say ".."
|
||||||
(%n if (%n > 0) else "No more")
|
|\%n if (%n > 0) else "No more"\ \"bottle" if (%n == 1) else "bottles"\ of beer on the wall.
|
||||||
(" bottle" if (%n == 1) else " bottles")
|
|\"" if (%n == 0) else " Take one down, pass it around..."\
|
||||||
" of beer on the wall."
|
|
||||||
("" if (%n == 0) else " Take one down, pass it around...")
|
|
||||||
|
|
||||||
sing 9 bottles of beer
|
sing 9 bottles of beer
|
||||||
|
|
||||||
@ -182,25 +187,23 @@ any of [0,0,0,0,1,0,0]
|
|||||||
rule "say the time":
|
rule "say the time":
|
||||||
lua block ".."
|
lua block ".."
|
||||||
|io.write("The OS time is: ")
|
|io.write("The OS time is: ")
|
||||||
|io.write(tostring(os.time()).."\n")
|
|io.write(tostring(os.time()).."\\n")
|
||||||
say the time
|
say the time
|
||||||
printf ["Math expression result is: ", lua expr "(1 + 2*3 + 3*4)^2"]
|
say ".."|Math expression result is: \lua expr "(1 + 2*3 + 3*4)^2"\
|
||||||
|
|
||||||
#.. In the lua environment, "vars" can be used to get local variables/function args, and
|
#.. In the lua environment, "vars" can be used to get local variables/function args, and
|
||||||
"compiler" can be used to access the compiler, function defs, and other things
|
"compiler" can be used to access the compiler, function defs, and other things
|
||||||
rule "square root of %n":
|
rule "square root of %n":
|
||||||
return (lua expr "math.sqrt(vars.n)")
|
return (lua expr "math.sqrt(vars.n)")
|
||||||
printf ["the square root of 2 is ", square root of 2]
|
say ".."|The square root of 2 is \square root of 2\
|
||||||
|
|
||||||
# Macros can be defined as functions that take unprocessed syntax trees and return lua code
|
# Macros can be defined as functions that take unprocessed syntax trees and return lua code
|
||||||
# "macro block %" is for defining macros that produce blocks of code, not values
|
# "macro block %" is for defining macros that produce blocks of code, not values
|
||||||
macro block "unless %condition %body":
|
macro block "unless %condition %body":
|
||||||
concat [..]
|
".."
|
||||||
# "% as lua expr" and "% as lua block" are two useful helper functions here.
|
|if not (\%condition as lua expr\) then
|
||||||
"if not (", %condition as lua expr, ") then"
|
| \(lua expr "vars.body.value.value") as lua block\
|
||||||
# Extract the inner part of the code block's body and insert it:
|
|end
|
||||||
"\n ", (lua expr "vars.body.value.value") as lua block
|
|
||||||
"\nend"
|
|
||||||
|
|
||||||
unless (1 > 10):
|
unless (1 > 10):
|
||||||
say "Macros work!"
|
say "Macros work!"
|
||||||
@ -208,6 +211,6 @@ unless (1 > 10):
|
|||||||
|
|
||||||
# and "macro %" is for defining macros that produce an expression
|
# and "macro %" is for defining macros that produce an expression
|
||||||
macro "%value as a boolean":
|
macro "%value as a boolean":
|
||||||
concat ["(not not (", %value as lua expr, "))"]
|
".."|(not not (\%value as lua expr\))
|
||||||
macro "yep": "true"
|
macro "yep": "true"
|
||||||
|
|
||||||
|
105
nomsu.lua
105
nomsu.lua
@ -59,7 +59,7 @@ make_parser = function(lingo, extra_definitions)
|
|||||||
end
|
end
|
||||||
return end_pos
|
return end_pos
|
||||||
end
|
end
|
||||||
local wordchar = P(1) - S(' \t\n\r%#:;,.{}[]()"')
|
local wordchar = P(1) - S(' \t\n\r%#:;,.{}[]()"\\')
|
||||||
local nl = P("\n")
|
local nl = P("\n")
|
||||||
local whitespace = S(" \t") ^ 1
|
local whitespace = S(" \t") ^ 1
|
||||||
local blank_line = whitespace ^ -1 * nl
|
local blank_line = whitespace ^ -1 * nl
|
||||||
@ -288,7 +288,17 @@ do
|
|||||||
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
|
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
|
||||||
|
|
||||||
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
||||||
longstring <- ({ '".."' %ws? %line_comment? %indent {(%new_line "|" [^%nl]*)+} ((%dedent (%new_line '..')?) / errors) }) -> Longstring
|
longstring <- ({ '".."' %ws?
|
||||||
|
{|
|
||||||
|
(("|" {| ({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)* |})
|
||||||
|
/ %line_comment)?
|
||||||
|
(%indent
|
||||||
|
(%new_line "|" {|
|
||||||
|
({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)*
|
||||||
|
|})+
|
||||||
|
((%dedent (%new_line '..')?) / errors))?
|
||||||
|
|}}) -> Longstring
|
||||||
|
string_interpolation <- "\" %ws? (functioncall / expression) %ws? "\"
|
||||||
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
|
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
|
||||||
variable <- ({ ("%" {%wordchar+}) }) -> Var
|
variable <- ({ ("%" {%wordchar+}) }) -> Var
|
||||||
|
|
||||||
@ -314,13 +324,13 @@ do
|
|||||||
assert(tree, "Failed to parse: " .. tostring(str))
|
assert(tree, "Failed to parse: " .. tostring(str))
|
||||||
return tree
|
return tree
|
||||||
end,
|
end,
|
||||||
tree_to_value = function(self, tree)
|
tree_to_value = function(self, tree, vars)
|
||||||
local code = "return (function(compiler, vars)\nreturn " .. tostring(self:tree_to_lua(tree)) .. "\nend)"
|
local code = "\n local utils = require('utils')\n return (function(compiler, vars)\nreturn " .. tostring(self:tree_to_lua(tree)) .. "\nend)"
|
||||||
local lua_thunk, err = load(code)
|
local lua_thunk, err = load(code)
|
||||||
if not lua_thunk then
|
if not lua_thunk then
|
||||||
error("Failed to compile generated code:\n" .. tostring(code) .. "\n\n" .. tostring(err))
|
error("Failed to compile generated code:\n" .. tostring(code) .. "\n\n" .. tostring(err))
|
||||||
end
|
end
|
||||||
return (lua_thunk())(self, { })
|
return (lua_thunk())(self, vars or { })
|
||||||
end,
|
end,
|
||||||
tree_to_lua = function(self, tree, kind)
|
tree_to_lua = function(self, tree, kind)
|
||||||
if kind == nil then
|
if kind == nil then
|
||||||
@ -347,9 +357,9 @@ do
|
|||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local statement = _list_0[_index_0]
|
local statement = _list_0[_index_0]
|
||||||
local code = to_lua(statement)
|
local code = to_lua(statement)
|
||||||
local lua_thunk, err = load("return (function(compiler, vars)\n" .. tostring(code) .. "\nend)")
|
local lua_thunk, err = load("\n local utils = require('utils')\n return (function(compiler, vars)\n" .. tostring(code) .. "\nend)")
|
||||||
if not lua_thunk then
|
if not lua_thunk then
|
||||||
error("Failed to compile generated code:\n" .. tostring(code) .. "\n\n" .. tostring(err))
|
error("Failed to compile generated code:\n" .. tostring(code) .. "\n\n" .. tostring(err) .. "\n\nProduced by statement:\n" .. tostring(utils.repr(statement)))
|
||||||
end
|
end
|
||||||
local ok
|
local ok
|
||||||
ok, err = pcall(lua_thunk)
|
ok, err = pcall(lua_thunk)
|
||||||
@ -429,17 +439,35 @@ do
|
|||||||
end))
|
end))
|
||||||
add(utils.repr(unescaped, true))
|
add(utils.repr(unescaped, true))
|
||||||
elseif "Longstring" == _exp_0 then
|
elseif "Longstring" == _exp_0 then
|
||||||
local result
|
local concat_parts = { }
|
||||||
do
|
local string_buffer = ""
|
||||||
local _accum_0 = { }
|
for i, line in ipairs(tree.value) do
|
||||||
local _len_0 = 1
|
if i > 1 then
|
||||||
for line in tree.value:gmatch("[ \t]*|([^\n]*)") do
|
string_buffer = string_buffer .. "\n"
|
||||||
_accum_0[_len_0] = line
|
end
|
||||||
_len_0 = _len_0 + 1
|
for _index_0 = 1, #line do
|
||||||
|
local bit = line[_index_0]
|
||||||
|
if type(bit) == "string" then
|
||||||
|
string_buffer = string_buffer .. bit:gsub("\\\\", "\\")
|
||||||
|
else
|
||||||
|
if string_buffer ~= "" then
|
||||||
|
table.insert(concat_parts, utils.repr(string_buffer, true))
|
||||||
|
string_buffer = ""
|
||||||
|
end
|
||||||
|
table.insert(concat_parts, "utils.repr(" .. tostring(to_lua(bit)) .. ")")
|
||||||
|
end
|
||||||
end
|
end
|
||||||
result = _accum_0
|
|
||||||
end
|
end
|
||||||
add(utils.repr(table.concat(result, "\n"), true))
|
if string_buffer ~= "" then
|
||||||
|
table.insert(concat_parts, utils.repr(string_buffer, true))
|
||||||
|
end
|
||||||
|
if #concat_parts == 0 then
|
||||||
|
add("''")
|
||||||
|
elseif #concat_parts == 1 then
|
||||||
|
add(concat_parts[1])
|
||||||
|
else
|
||||||
|
add("(" .. tostring(table.concat(concat_parts, "..")) .. ")")
|
||||||
|
end
|
||||||
elseif "Number" == _exp_0 then
|
elseif "Number" == _exp_0 then
|
||||||
add(tree.value)
|
add(tree.value)
|
||||||
elseif "List" == _exp_0 then
|
elseif "List" == _exp_0 then
|
||||||
@ -670,34 +698,12 @@ do
|
|||||||
end,
|
end,
|
||||||
initialize_core = function(self)
|
initialize_core = function(self)
|
||||||
local as_lua_code
|
local as_lua_code
|
||||||
as_lua_code = function(self, str)
|
as_lua_code = function(self, str, vars)
|
||||||
local _exp_0 = str.type
|
local _exp_0 = str.type
|
||||||
if "String" == _exp_0 then
|
if "String" == _exp_0 then
|
||||||
local escapes = {
|
return self:tree_to_value(str, vars)
|
||||||
n = "\n",
|
|
||||||
t = "\t",
|
|
||||||
b = "\b",
|
|
||||||
a = "\a",
|
|
||||||
v = "\v",
|
|
||||||
f = "\f",
|
|
||||||
r = "\r"
|
|
||||||
}
|
|
||||||
local unescaped = str.value:gsub("\\(.)", (function(c)
|
|
||||||
return escapes[c] or c
|
|
||||||
end))
|
|
||||||
return unescaped
|
|
||||||
elseif "Longstring" == _exp_0 then
|
elseif "Longstring" == _exp_0 then
|
||||||
local result
|
return self:tree_to_value(str, vars)
|
||||||
do
|
|
||||||
local _accum_0 = { }
|
|
||||||
local _len_0 = 1
|
|
||||||
for line in str.value:gmatch("[ \t]*|([^\n]*)") do
|
|
||||||
_accum_0[_len_0] = line
|
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
|
||||||
result = _accum_0
|
|
||||||
end
|
|
||||||
return table.concat(result, "\n")
|
|
||||||
else
|
else
|
||||||
return self:tree_to_lua(str)
|
return self:tree_to_lua(str)
|
||||||
end
|
end
|
||||||
@ -715,13 +721,13 @@ do
|
|||||||
local _list_0 = lua_code.value
|
local _list_0 = lua_code.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local i = _list_0[_index_0]
|
local i = _list_0[_index_0]
|
||||||
_accum_0[_len_0] = as_lua_code(self, i.value)
|
_accum_0[_len_0] = as_lua_code(self, i.value, vars)
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
return _accum_0
|
return _accum_0
|
||||||
end)()), true
|
end)()), true
|
||||||
else
|
else
|
||||||
return as_lua_code(self, lua_code), true
|
return as_lua_code(self, lua_code, vars), true
|
||||||
end
|
end
|
||||||
end)
|
end)
|
||||||
self:defmacro([[lua expr %lua_code]], function(self, vars, kind)
|
self:defmacro([[lua expr %lua_code]], function(self, vars, kind)
|
||||||
@ -734,13 +740,13 @@ do
|
|||||||
local _list_0 = lua_code.value
|
local _list_0 = lua_code.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local i = _list_0[_index_0]
|
local i = _list_0[_index_0]
|
||||||
_accum_0[_len_0] = as_lua_code(self, i.value)
|
_accum_0[_len_0] = as_lua_code(self, i.value, vars)
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
return _accum_0
|
return _accum_0
|
||||||
end)())
|
end)())
|
||||||
else
|
else
|
||||||
return as_lua_code(self, lua_code)
|
return as_lua_code(self, lua_code, vars)
|
||||||
end
|
end
|
||||||
end)
|
end)
|
||||||
self:def("rule %spec %body", function(self, vars)
|
self:def("rule %spec %body", function(self, vars)
|
||||||
@ -750,15 +756,15 @@ do
|
|||||||
if kind == "Expression" then
|
if kind == "Expression" then
|
||||||
error("Macro definitions cannot be used as expressions.")
|
error("Macro definitions cannot be used as expressions.")
|
||||||
end
|
end
|
||||||
self:defmacro(self:tree_to_value(vars.spec), self:tree_to_value(vars.body))
|
self:defmacro(self:tree_to_value(vars.spec, vars), self:tree_to_value(vars.body, vars))
|
||||||
return "", true
|
return "", true
|
||||||
end)
|
end)
|
||||||
self:defmacro([[macro block %spec %body]], function(self, vars, kind)
|
self:defmacro([[macro block %spec %body]], function(self, vars, kind)
|
||||||
if kind == "Expression" then
|
if kind == "Expression" then
|
||||||
error("Macro definitions cannot be used as expressions.")
|
error("Macro definitions cannot be used as expressions.")
|
||||||
end
|
end
|
||||||
local invocation = self:tree_to_value(vars.spec)
|
local invocation = self:tree_to_value(vars.spec, vars)
|
||||||
local fn = self:tree_to_value(vars.body)
|
local fn = self:tree_to_value(vars.body, vars)
|
||||||
self:defmacro(invocation, (function(self, vars, kind)
|
self:defmacro(invocation, (function(self, vars, kind)
|
||||||
if kind == "Expression" then
|
if kind == "Expression" then
|
||||||
error("Macro: " .. tostring(invocation) .. " was defined to be a block, not an expression.")
|
error("Macro: " .. tostring(invocation) .. " was defined to be a block, not an expression.")
|
||||||
@ -837,7 +843,8 @@ if arg and arg[1] then
|
|||||||
else
|
else
|
||||||
output = io.open(arg[2], 'w')
|
output = io.open(arg[2], 'w')
|
||||||
end
|
end
|
||||||
output:write([[ local load = function()
|
output:write([[ local utils = require('utils')
|
||||||
|
local load = function()
|
||||||
]])
|
]])
|
||||||
output:write(code)
|
output:write(code)
|
||||||
output:write([[
|
output:write([[
|
||||||
|
83
nomsu.moon
83
nomsu.moon
@ -4,7 +4,6 @@ lpeg = require 'lpeg'
|
|||||||
utils = require 'utils'
|
utils = require 'utils'
|
||||||
|
|
||||||
-- TODO:
|
-- TODO:
|
||||||
-- string interpolation
|
|
||||||
-- improve indentation of generated lua code
|
-- improve indentation of generated lua code
|
||||||
-- provide way to run precompiled nomsu -> lua code
|
-- provide way to run precompiled nomsu -> lua code
|
||||||
-- comprehensions?
|
-- comprehensions?
|
||||||
@ -44,7 +43,7 @@ make_parser = (lingo, extra_definitions)->
|
|||||||
if num_spaces != indent_stack[#indent_stack] then return nil
|
if num_spaces != indent_stack[#indent_stack] then return nil
|
||||||
return end_pos
|
return end_pos
|
||||||
|
|
||||||
wordchar = P(1)-S(' \t\n\r%#:;,.{}[]()"')
|
wordchar = P(1)-S(' \t\n\r%#:;,.{}[]()"\\')
|
||||||
nl = P("\n")
|
nl = P("\n")
|
||||||
whitespace = S(" \t")^1
|
whitespace = S(" \t")^1
|
||||||
blank_line = whitespace^-1 * nl
|
blank_line = whitespace^-1 * nl
|
||||||
@ -199,7 +198,17 @@ class NomsuCompiler
|
|||||||
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
|
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
|
||||||
|
|
||||||
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
||||||
longstring <- ({ '".."' %ws? %line_comment? %indent {(%new_line "|" [^%nl]*)+} ((%dedent (%new_line '..')?) / errors) }) -> Longstring
|
longstring <- ({ '".."' %ws?
|
||||||
|
{|
|
||||||
|
(("|" {| ({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)* |})
|
||||||
|
/ %line_comment)?
|
||||||
|
(%indent
|
||||||
|
(%new_line "|" {|
|
||||||
|
({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)*
|
||||||
|
|})+
|
||||||
|
((%dedent (%new_line '..')?) / errors))?
|
||||||
|
|}}) -> Longstring
|
||||||
|
string_interpolation <- "\" %ws? (functioncall / expression) %ws? "\"
|
||||||
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
|
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
|
||||||
variable <- ({ ("%" {%wordchar+}) }) -> Var
|
variable <- ({ ("%" {%wordchar+}) }) -> Var
|
||||||
|
|
||||||
@ -225,12 +234,15 @@ class NomsuCompiler
|
|||||||
assert tree, "Failed to parse: #{str}"
|
assert tree, "Failed to parse: #{str}"
|
||||||
return tree
|
return tree
|
||||||
|
|
||||||
tree_to_value: (tree)=>
|
tree_to_value: (tree, vars)=>
|
||||||
code = "return (function(compiler, vars)\nreturn #{@tree_to_lua(tree)}\nend)"
|
-- TODO: clean up require utils
|
||||||
|
code = "
|
||||||
|
local utils = require('utils')
|
||||||
|
return (function(compiler, vars)\nreturn #{@tree_to_lua(tree)}\nend)"
|
||||||
lua_thunk, err = load(code)
|
lua_thunk, err = load(code)
|
||||||
if not lua_thunk
|
if not lua_thunk
|
||||||
error("Failed to compile generated code:\n#{code}\n\n#{err}")
|
error("Failed to compile generated code:\n#{code}\n\n#{err}")
|
||||||
return (lua_thunk!)(self, {})
|
return (lua_thunk!)(self, vars or {})
|
||||||
|
|
||||||
tree_to_lua: (tree, kind="Expression")=>
|
tree_to_lua: (tree, kind="Expression")=>
|
||||||
assert tree, "No tree provided."
|
assert tree, "No tree provided."
|
||||||
@ -251,9 +263,12 @@ class NomsuCompiler
|
|||||||
for statement in *tree.value.body.value
|
for statement in *tree.value.body.value
|
||||||
code = to_lua(statement)
|
code = to_lua(statement)
|
||||||
-- Run the fuckers as we go
|
-- Run the fuckers as we go
|
||||||
lua_thunk, err = load("return (function(compiler, vars)\n#{code}\nend)")
|
-- TODO: clean up repeated loading of utils?
|
||||||
|
lua_thunk, err = load("
|
||||||
|
local utils = require('utils')
|
||||||
|
return (function(compiler, vars)\n#{code}\nend)")
|
||||||
if not lua_thunk
|
if not lua_thunk
|
||||||
error("Failed to compile generated code:\n#{code}\n\n#{err}")
|
error("Failed to compile generated code:\n#{code}\n\n#{err}\n\nProduced by statement:\n#{utils.repr(statement)}")
|
||||||
ok,err = pcall(lua_thunk)
|
ok,err = pcall(lua_thunk)
|
||||||
if not ok then error(err)
|
if not ok then error(err)
|
||||||
ok,err = pcall(err, self, vars)
|
ok,err = pcall(err, self, vars)
|
||||||
@ -308,9 +323,28 @@ class NomsuCompiler
|
|||||||
add utils.repr(unescaped, true)
|
add utils.repr(unescaped, true)
|
||||||
|
|
||||||
when "Longstring"
|
when "Longstring"
|
||||||
-- TODO: handle comments here?
|
concat_parts = {}
|
||||||
result = [line for line in tree.value\gmatch("[ \t]*|([^\n]*)")]
|
string_buffer = ""
|
||||||
add utils.repr(table.concat(result, "\n"), true)
|
for i,line in ipairs(tree.value)
|
||||||
|
if i > 1 then string_buffer ..= "\n"
|
||||||
|
for bit in *line
|
||||||
|
if type(bit) == "string"
|
||||||
|
string_buffer ..= bit\gsub("\\\\","\\")
|
||||||
|
else
|
||||||
|
if string_buffer ~= ""
|
||||||
|
table.insert concat_parts, utils.repr(string_buffer, true)
|
||||||
|
string_buffer = ""
|
||||||
|
table.insert concat_parts, "utils.repr(#{to_lua(bit)})"
|
||||||
|
|
||||||
|
if string_buffer ~= ""
|
||||||
|
table.insert concat_parts, utils.repr(string_buffer, true)
|
||||||
|
|
||||||
|
if #concat_parts == 0
|
||||||
|
add "''"
|
||||||
|
elseif #concat_parts == 1
|
||||||
|
add concat_parts[1]
|
||||||
|
else
|
||||||
|
add "(#{table.concat(concat_parts, "..")})"
|
||||||
|
|
||||||
when "Number"
|
when "Number"
|
||||||
add tree.value
|
add tree.value
|
||||||
@ -482,17 +516,12 @@ class NomsuCompiler
|
|||||||
|
|
||||||
initialize_core: =>
|
initialize_core: =>
|
||||||
-- Sets up some core functionality
|
-- Sets up some core functionality
|
||||||
as_lua_code = (str)=>
|
as_lua_code = (str, vars)=>
|
||||||
switch str.type
|
switch str.type
|
||||||
when "String"
|
when "String"
|
||||||
escapes = n:"\n", t:"\t", b:"\b", a:"\a", v:"\v", f:"\f", r:"\r"
|
return @tree_to_value(str, vars)
|
||||||
unescaped = str.value\gsub("\\(.)", ((c)-> escapes[c] or c))
|
|
||||||
return unescaped
|
|
||||||
|
|
||||||
when "Longstring"
|
when "Longstring"
|
||||||
-- TODO: handle comments?
|
return @tree_to_value(str, vars)
|
||||||
result = [line for line in str.value\gmatch("[ \t]*|([^\n]*)")]
|
|
||||||
return table.concat(result, "\n")
|
|
||||||
else
|
else
|
||||||
return @tree_to_lua(str)
|
return @tree_to_lua(str)
|
||||||
|
|
||||||
@ -502,18 +531,18 @@ class NomsuCompiler
|
|||||||
switch lua_code.type
|
switch lua_code.type
|
||||||
when "List"
|
when "List"
|
||||||
-- TODO: handle subexpressions
|
-- TODO: handle subexpressions
|
||||||
return table.concat([as_lua_code(@, i.value) for i in *lua_code.value]), true
|
return table.concat([as_lua_code(@, i.value, vars) for i in *lua_code.value]), true
|
||||||
else
|
else
|
||||||
return as_lua_code(@, lua_code), true
|
return as_lua_code(@, lua_code, vars), true
|
||||||
|
|
||||||
@defmacro [[lua expr %lua_code]], (vars, kind)=>
|
@defmacro [[lua expr %lua_code]], (vars, kind)=>
|
||||||
lua_code = vars.lua_code.value
|
lua_code = vars.lua_code.value
|
||||||
switch lua_code.type
|
switch lua_code.type
|
||||||
when "List"
|
when "List"
|
||||||
-- TODO: handle subexpressions
|
-- TODO: handle subexpressions
|
||||||
return table.concat([as_lua_code(@, i.value) for i in *lua_code.value])
|
return table.concat([as_lua_code(@, i.value, vars) for i in *lua_code.value])
|
||||||
else
|
else
|
||||||
return as_lua_code(@, lua_code)
|
return as_lua_code(@, lua_code, vars)
|
||||||
|
|
||||||
@def "rule %spec %body", (vars)=>
|
@def "rule %spec %body", (vars)=>
|
||||||
@def vars.spec, vars.body
|
@def vars.spec, vars.body
|
||||||
@ -521,14 +550,14 @@ class NomsuCompiler
|
|||||||
@defmacro [[macro %spec %body]], (vars, kind)=>
|
@defmacro [[macro %spec %body]], (vars, kind)=>
|
||||||
if kind == "Expression"
|
if kind == "Expression"
|
||||||
error("Macro definitions cannot be used as expressions.")
|
error("Macro definitions cannot be used as expressions.")
|
||||||
@defmacro @tree_to_value(vars.spec), @tree_to_value(vars.body)
|
@defmacro @tree_to_value(vars.spec, vars), @tree_to_value(vars.body, vars)
|
||||||
return "", true
|
return "", true
|
||||||
|
|
||||||
@defmacro [[macro block %spec %body]], (vars, kind)=>
|
@defmacro [[macro block %spec %body]], (vars, kind)=>
|
||||||
if kind == "Expression"
|
if kind == "Expression"
|
||||||
error("Macro definitions cannot be used as expressions.")
|
error("Macro definitions cannot be used as expressions.")
|
||||||
invocation = @tree_to_value(vars.spec)
|
invocation = @tree_to_value(vars.spec, vars)
|
||||||
fn = @tree_to_value(vars.body)
|
fn = @tree_to_value(vars.body, vars)
|
||||||
@defmacro invocation, ((vars,kind)=>
|
@defmacro invocation, ((vars,kind)=>
|
||||||
if kind == "Expression"
|
if kind == "Expression"
|
||||||
error("Macro: #{invocation} was defined to be a block, not an expression.")
|
error("Macro: #{invocation} was defined to be a block, not an expression.")
|
||||||
@ -544,6 +573,7 @@ class NomsuCompiler
|
|||||||
-- and "./nomsu.moon input_file.nom output_file.lua" to compile (use "-" to compile to stdout)
|
-- and "./nomsu.moon input_file.nom output_file.lua" to compile (use "-" to compile to stdout)
|
||||||
if arg and arg[1]
|
if arg and arg[1]
|
||||||
c = NomsuCompiler()
|
c = NomsuCompiler()
|
||||||
|
--c.debug = true
|
||||||
input = io.open(arg[1])\read("*a")
|
input = io.open(arg[1])\read("*a")
|
||||||
-- Kinda hacky, if run via "./nomsu.moon file.nom -", then silence print and io.write
|
-- Kinda hacky, if run via "./nomsu.moon file.nom -", then silence print and io.write
|
||||||
-- during execution and re-enable them to print out the generated source code
|
-- during execution and re-enable them to print out the generated source code
|
||||||
@ -562,6 +592,7 @@ if arg and arg[1]
|
|||||||
else io.open(arg[2], 'w')
|
else io.open(arg[2], 'w')
|
||||||
|
|
||||||
output\write [[
|
output\write [[
|
||||||
|
local utils = require('utils')
|
||||||
local load = function()
|
local load = function()
|
||||||
]]
|
]]
|
||||||
output\write(code)
|
output\write(code)
|
||||||
|
Loading…
Reference in New Issue
Block a user