Working string interpolation!

This commit is contained in:
Bruce Hill 2017-09-14 02:41:10 -07:00
parent 0f228d2d25
commit 83463f11c5
6 changed files with 156 additions and 116 deletions

View File

@ -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)"

View File

@ -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 [..]

View File

@ -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

View File

@ -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
View File

@ -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([[

View File

@ -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)