Much better error reporting for compile errors (i.e. not parse errors),
using the pretty_error system.
This commit is contained in:
parent
96e5e567cb
commit
f225a48367
@ -197,7 +197,7 @@ compile [..]
|
|||||||
..to:
|
..to:
|
||||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||||
unless (%var.type is "Var"):
|
unless (%var.type is "Var"):
|
||||||
compile error at %var.source "Loop expected variable, not: %s"
|
compile error at %var "Expected a variable here, not a \(%var.type)."
|
||||||
%lua = (..)
|
%lua = (..)
|
||||||
Lua "\
|
Lua "\
|
||||||
..for \(%var as lua expr)=\(%start as lua expr),\(%stop as lua expr),\(..)
|
..for \(%var as lua expr)=\(%start as lua expr),\(%stop as lua expr),\(..)
|
||||||
@ -242,7 +242,7 @@ test:
|
|||||||
compile [for %var in %iterable %body] to:
|
compile [for %var in %iterable %body] to:
|
||||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||||
unless (%var.type is "Var"):
|
unless (%var.type is "Var"):
|
||||||
compile error at %var.source "Loop expected variable, not: %s"
|
compile error at %var "Expected a variable here, not a \(%var.type)."
|
||||||
define mangler
|
define mangler
|
||||||
%lua = (..)
|
%lua = (..)
|
||||||
Lua "\
|
Lua "\
|
||||||
@ -280,9 +280,9 @@ compile [..]
|
|||||||
..to:
|
..to:
|
||||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||||
unless (%key.type is "Var"):
|
unless (%key.type is "Var"):
|
||||||
compile error at %key.source "Loop expected variable, not: %s"
|
compile error at %key "Expected a variable here, not a \(%key.type)."
|
||||||
unless (%value.type is "Var"):
|
unless (%value.type is "Var"):
|
||||||
compile error at %value.source "Loop expected variable, not: %s"
|
compile error at %value "Expected a variable here, not a \(%value.type)."
|
||||||
%lua = (..)
|
%lua = (..)
|
||||||
Lua "\
|
Lua "\
|
||||||
..for \(%key as lua identifier),\(%value as lua identifier) in pairs(\(..)
|
..for \(%key as lua identifier),\(%value as lua identifier) in pairs(\(..)
|
||||||
@ -331,23 +331,28 @@ compile [if %body, when %body] to:
|
|||||||
%clause = "if"
|
%clause = "if"
|
||||||
%else_allowed = (yes)
|
%else_allowed = (yes)
|
||||||
unless (%body.type is "Block"):
|
unless (%body.type is "Block"):
|
||||||
compile error at %body.source "'if' expected a Block, but got: %s"
|
compile error at %body "'if' expected a Block, but got a \(%body.type)."
|
||||||
|
..hint "Perhaps you forgot to put a ':' after 'if'?"
|
||||||
for %line in %body:
|
for %line in %body:
|
||||||
unless (..)
|
unless (..)
|
||||||
((%line.type is "Action") and ((size of %line) >= 2)) and (..)
|
((%line.type is "Action") and ((size of %line) >= 2)) and (..)
|
||||||
%line.(size of %line) is "Block" syntax tree
|
%line.(size of %line) is "Block" syntax tree
|
||||||
..:
|
..:
|
||||||
compile error at %line.source "\
|
compile error at %line "Invalid line for the body of an 'if' block."
|
||||||
..Invalid line for 'if', each line should contain conditional expressions followed by a block, or "else" followed by a block:
|
..hint "Each line should contain one or more conditional expressions \
|
||||||
%s"
|
..followed by a block, or "else" followed by a block."
|
||||||
|
|
||||||
%action = %line.(size of %line)
|
%action = %line.(size of %line)
|
||||||
if ((%line.1 is "else") and ((size of %line) == 2)):
|
if ((%line.1 is "else") and ((size of %line) == 2)):
|
||||||
unless %else_allowed:
|
unless %else_allowed:
|
||||||
compile error at %line.source "Can't have two 'else' blocks"
|
compile error at %line "You can't have two 'else' blocks."
|
||||||
|
..hint "Merge all of the 'else' blocks together."
|
||||||
unless ((size of "\%code") > 0):
|
unless ((size of "\%code") > 0):
|
||||||
compile error at %line.source "\
|
compile error at %line "\
|
||||||
..Can't have an 'else' block without a preceeding condition"
|
..You can't have an 'else' block without a preceeding condition"
|
||||||
|
..hint "If you want the code in this block to always execute, you don't \
|
||||||
|
..need a conditional block around it. Otherwise, make sure the 'else' \
|
||||||
|
..block comes last."
|
||||||
|
|
||||||
%code::append "\
|
%code::append "\
|
||||||
..
|
..
|
||||||
@ -358,11 +363,6 @@ compile [if %body, when %body] to:
|
|||||||
..else:
|
..else:
|
||||||
%code::append "\%clause "
|
%code::append "\%clause "
|
||||||
for %i in 1 to ((size of %line) - 1):
|
for %i in 1 to ((size of %line) - 1):
|
||||||
unless (%line.%i is syntax tree):
|
|
||||||
compile error at %line.source "\
|
|
||||||
..Invalid condition for 'if' statement:
|
|
||||||
%s"
|
|
||||||
|
|
||||||
if (%i > 1):
|
if (%i > 1):
|
||||||
%code::append " or "
|
%code::append " or "
|
||||||
%code::append (%line.%i as lua expr)
|
%code::append (%line.%i as lua expr)
|
||||||
@ -374,7 +374,8 @@ compile [if %body, when %body] to:
|
|||||||
%clause = "\nelseif"
|
%clause = "\nelseif"
|
||||||
|
|
||||||
if ((size of "\%code") == 0):
|
if ((size of "\%code") == 0):
|
||||||
compile error at %body.source "'if' block has an empty body"
|
compile error at %body "'if' block has an empty body."
|
||||||
|
..hint "This means nothing would happen, so the 'if' block should be deleted."
|
||||||
%code::append "\nend --when"
|
%code::append "\nend --when"
|
||||||
return %code
|
return %code
|
||||||
|
|
||||||
@ -395,23 +396,28 @@ compile [if %branch_value is %body, when %branch_value is %body] to:
|
|||||||
%else_allowed = (yes)
|
%else_allowed = (yes)
|
||||||
define mangler
|
define mangler
|
||||||
unless (%body.type is "Block"):
|
unless (%body.type is "Block"):
|
||||||
compile error at %body.source "'if' expected a Block, but got: %s"
|
compile error at %body "'if' expected a Block, but got a \(%body.type)"
|
||||||
|
..hint "Perhaps you forgot to put a ':' after the 'is'?"
|
||||||
for %line in %body:
|
for %line in %body:
|
||||||
unless (..)
|
unless (..)
|
||||||
((%line.type is "Action") and ((size of %line) >= 2)) and (..)
|
((%line.type is "Action") and ((size of %line) >= 2)) and (..)
|
||||||
%line.(size of %line) is "Block" syntax tree
|
%line.(size of %line) is "Block" syntax tree
|
||||||
..:
|
..:
|
||||||
compile error at %line.source "\
|
compile error at %line "Invalid line for 'if' block."
|
||||||
..Invalid line for 'if % is % %', each line should contain expressions followed by a block, or "else" followed by a block:
|
..hint "Each line should contain expressions \
|
||||||
%s"
|
..followed by a block, or "else" followed by a block"
|
||||||
|
|
||||||
%action = %line.(size of %line)
|
%action = %line.(size of %line)
|
||||||
if ((%line.1 is "else") and ((size of %line) == 2)):
|
if ((%line.1 is "else") and ((size of %line) == 2)):
|
||||||
unless %else_allowed:
|
unless %else_allowed:
|
||||||
compile error at %line.source "Can't have two 'else' blocks"
|
compile error at %line "You can't have two 'else' blocks."
|
||||||
|
..hint "Merge all of the 'else' blocks together."
|
||||||
unless ((size of "\%code") > 0):
|
unless ((size of "\%code") > 0):
|
||||||
compile error at %line.source "\
|
compile error at %line "\
|
||||||
..Can't have an 'else' block without a preceeding condition"
|
..You can't have an 'else' block without a preceeding condition"
|
||||||
|
..hint "If you want the code in this block to always execute, you don't \
|
||||||
|
..need a conditional block around it. Otherwise, make sure the 'else' \
|
||||||
|
..block comes last."
|
||||||
|
|
||||||
%code::append "\
|
%code::append "\
|
||||||
..
|
..
|
||||||
@ -422,11 +428,6 @@ compile [if %branch_value is %body, when %branch_value is %body] to:
|
|||||||
..else:
|
..else:
|
||||||
%code::append "\%clause "
|
%code::append "\%clause "
|
||||||
for %i in 1 to ((size of %line) - 1):
|
for %i in 1 to ((size of %line) - 1):
|
||||||
unless (%line.%i is syntax tree):
|
|
||||||
compile error at %line.source "\
|
|
||||||
..Invalid condition for 'if' statement:
|
|
||||||
%s"
|
|
||||||
|
|
||||||
if (%i > 1):
|
if (%i > 1):
|
||||||
%code::append " or "
|
%code::append " or "
|
||||||
%code::append "\(mangle "branch value") == \(%line.%i as lua expr)"
|
%code::append "\(mangle "branch value") == \(%line.%i as lua expr)"
|
||||||
@ -438,7 +439,8 @@ compile [if %branch_value is %body, when %branch_value is %body] to:
|
|||||||
%clause = "\nelseif"
|
%clause = "\nelseif"
|
||||||
|
|
||||||
if ((size of "\%code") == 0):
|
if ((size of "\%code") == 0):
|
||||||
compile error at %body.source "'if % is % %' block has an empty body"
|
compile error at %body "'if' block has an empty body."
|
||||||
|
..hint "This means nothing would happen, so the 'if' block should be deleted."
|
||||||
%code::append "\nend --when"
|
%code::append "\nend --when"
|
||||||
return (..)
|
return (..)
|
||||||
Lua "\
|
Lua "\
|
||||||
|
@ -6,8 +6,10 @@ use "core/metaprogramming.nom"
|
|||||||
|
|
||||||
compile [barf] to (Lua "error(nil, 0);")
|
compile [barf] to (Lua "error(nil, 0);")
|
||||||
compile [barf %msg] to (Lua "error(\(%msg as lua expr), 0);")
|
compile [barf %msg] to (Lua "error(\(%msg as lua expr), 0);")
|
||||||
compile [compile error at %source %msg] to (..)
|
compile [compile error at %tree %msg] to (..)
|
||||||
Lua "nomsu:compile_error(\(%source as lua expr), \(%msg as lua expr))"
|
Lua "nomsu:compile_error(\(%tree as lua expr), \(%msg as lua expr))"
|
||||||
|
compile [compile error at %tree %msg hint %hint] to (..)
|
||||||
|
Lua "nomsu:compile_error(\(%tree as lua expr), \(%msg as lua expr), \(%hint as lua expr))"
|
||||||
|
|
||||||
compile [assume %condition] to:
|
compile [assume %condition] to:
|
||||||
lua> "\
|
lua> "\
|
||||||
|
@ -36,7 +36,7 @@ compile [with local %locals %body, with local %locals do %body] to:
|
|||||||
"Var" "Action":
|
"Var" "Action":
|
||||||
%body_lua::declare locals ["\(%locals as lua)"]
|
%body_lua::declare locals ["\(%locals as lua)"]
|
||||||
else:
|
else:
|
||||||
compile error at %locals.source "Unexpected locals: %s"
|
compile error at %locals "Unexpected local value"
|
||||||
|
|
||||||
return (..)
|
return (..)
|
||||||
Lua "\
|
Lua "\
|
||||||
|
@ -322,11 +322,13 @@ do
|
|||||||
for i, t in ipairs(errs) do
|
for i, t in ipairs(errs) do
|
||||||
if i <= 3 then
|
if i <= 3 then
|
||||||
_accum_0[_len_0] = pretty_error({
|
_accum_0[_len_0] = pretty_error({
|
||||||
|
title = "Parse error",
|
||||||
error = t.error,
|
error = t.error,
|
||||||
hint = t.hint,
|
hint = t.hint,
|
||||||
source = t:get_source_code(),
|
source = t:get_source_code(),
|
||||||
start = t.source.start,
|
start = t.source.start,
|
||||||
stop = t.source.stop
|
stop = t.source.stop,
|
||||||
|
filename = t.source.filename
|
||||||
})
|
})
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
@ -340,19 +342,20 @@ do
|
|||||||
end
|
end
|
||||||
return tree
|
return tree
|
||||||
end
|
end
|
||||||
NomsuCompiler.compile_error = function(self, source, err_format_string, ...)
|
NomsuCompiler.compile_error = function(self, tree, err_msg, hint)
|
||||||
err_format_string = err_format_string:gsub("%%[^s]", "%%%1")
|
if hint == nil then
|
||||||
local file = Files.read(source.filename)
|
hint = nil
|
||||||
local line_starts = Files.get_line_starts(file)
|
end
|
||||||
local line_no = Files.get_line_number(file, source.start)
|
local err_str = pretty_error({
|
||||||
local line_start = line_starts[line_no]
|
title = "Compile error",
|
||||||
local src = colored.dim(file:sub(line_start, source.start - 1))
|
error = err_msg,
|
||||||
src = src .. colored.underscore(colored.bright(colored.red(file:sub(source.start, source.stop - 1))))
|
hint = hint,
|
||||||
local end_of_line = (line_starts[Files.get_line_number(file, source.stop) + 1] or 0) - 1
|
source = tree:get_source_code(),
|
||||||
src = src .. colored.dim(file:sub(source.stop, end_of_line - 1))
|
start = tree.source.start,
|
||||||
src = ' ' .. src:gsub('\n', '\n ')
|
stop = tree.source.stop,
|
||||||
local err_msg = err_format_string:format(src, ...)
|
filename = tree.source.filename
|
||||||
return error(tostring(source.filename) .. ":" .. tostring(line_no) .. ": " .. err_msg, 0)
|
})
|
||||||
|
return error(err_str, 0)
|
||||||
end
|
end
|
||||||
local add_lua_bits
|
local add_lua_bits
|
||||||
add_lua_bits = function(self, val_or_stmt, code, compile_actions)
|
add_lua_bits = function(self, val_or_stmt, code, compile_actions)
|
||||||
@ -369,7 +372,7 @@ do
|
|||||||
else
|
else
|
||||||
local bit_lua = self:compile(bit, compile_actions)
|
local bit_lua = self:compile(bit, compile_actions)
|
||||||
if not (bit_lua.is_value) then
|
if not (bit_lua.is_value) then
|
||||||
self:compile_error(bit.source, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.")
|
self:compile_error(bit, "Can't use this as a string interpolation value, since it's not an expression.")
|
||||||
end
|
end
|
||||||
lua:append(bit_lua)
|
lua:append(bit_lua)
|
||||||
end
|
end
|
||||||
@ -402,7 +405,7 @@ do
|
|||||||
else
|
else
|
||||||
local bit_lua = self:compile(bit)
|
local bit_lua = self:compile(bit)
|
||||||
if not (bit_lua.is_value) then
|
if not (bit_lua.is_value) then
|
||||||
self:compile_error(bit.source, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.")
|
self:compile_error(bit, "Can't use this as a string interpolation value, since it's not an expression.")
|
||||||
end
|
end
|
||||||
add_bit_lua(lua, bit_lua)
|
add_bit_lua(lua, bit_lua)
|
||||||
end
|
end
|
||||||
@ -422,7 +425,7 @@ do
|
|||||||
else
|
else
|
||||||
local tok_lua = self:compile(tok, compile_actions)
|
local tok_lua = self:compile(tok, compile_actions)
|
||||||
if not (tok_lua.is_value) then
|
if not (tok_lua.is_value) then
|
||||||
self:compile_error(tok.source, "Non-expression value inside math expression:\n%s")
|
self:compile_error(tok, "Can't use this as a value in a math expression, since it's not a value.")
|
||||||
end
|
end
|
||||||
if tok.type == "Action" then
|
if tok.type == "Action" then
|
||||||
tok_lua:parenthesize()
|
tok_lua:parenthesize()
|
||||||
@ -700,7 +703,8 @@ do
|
|||||||
end
|
end
|
||||||
local ret = compile_action(self, tree, unpack(args))
|
local ret = compile_action(self, tree, unpack(args))
|
||||||
if not ret then
|
if not ret then
|
||||||
self:compile_error(tree.source, "Compile-time action:\n%s\nfailed to produce any Lua")
|
local info = debug.getinfo(compile_action, "S")
|
||||||
|
self:compile_error(tree, "The compile-time action here (" .. tostring(stub) .. ") failed to produce any Lua", "Look at the implementation of (" .. tostring(stub) .. ") in " .. tostring(info.short_src:sub(1, 200)) .. ":" .. tostring(info.linedefined) .. " and make sure it's returning Lua code.")
|
||||||
end
|
end
|
||||||
return ret
|
return ret
|
||||||
end
|
end
|
||||||
@ -726,9 +730,11 @@ do
|
|||||||
local arg_lua = self:compile(tok, compile_actions)
|
local arg_lua = self:compile(tok, compile_actions)
|
||||||
if not (arg_lua.is_value) then
|
if not (arg_lua.is_value) then
|
||||||
if tok.type == "Block" then
|
if tok.type == "Block" then
|
||||||
self:compile_error(tok.source, ("Cannot compile action '" .. tostring(stub) .. "' with a Block as an argument.\n" .. "Maybe there should be a compile-time action with that name that isn't being found?"))
|
self:compile_error(tok, "Can't compile action (" .. tostring(stub) .. ") with a Block as an argument.", "Maybe there should be a compile-time action with that name that isn't being found?")
|
||||||
|
elseif tok.type == "Action" then
|
||||||
|
self:compile_error(tok, "Can't use this as an argument to (" .. tostring(stub) .. "), since it's not an expression, it produces: " .. tostring(repr(arg_lua)), "Check the implementation of (" .. tostring(tok.stub) .. ") to see if it is actually meant to produce an expression.")
|
||||||
else
|
else
|
||||||
self:compile_error(tok.source, "Cannot use:\n%s\nas an argument to '%s', since it's not an expression, it produces: %s", stub, repr(arg_lua))
|
self:compile_error(tok, "Can't use this as an argument to (" .. tostring(stub) .. "), since it's not an expression, it produces: " .. tostring(repr(arg_lua)))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
insert(args, arg_lua)
|
insert(args, arg_lua)
|
||||||
@ -800,7 +806,7 @@ do
|
|||||||
if not (bit_lua.is_value) then
|
if not (bit_lua.is_value) then
|
||||||
local src = ' ' .. gsub(tostring(self:compile(bit, compile_actions)), '\n', '\n ')
|
local src = ' ' .. gsub(tostring(self:compile(bit, compile_actions)), '\n', '\n ')
|
||||||
local line = tostring(bit.source.filename) .. ":" .. tostring(Files.get_line_number(Files.read(bit.source.filename), bit.source.start))
|
local line = tostring(bit.source.filename) .. ":" .. tostring(Files.get_line_number(Files.read(bit.source.filename), bit.source.start))
|
||||||
self:compile_error(bit.source, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.")
|
self:compile_error(bit, "Can't this as a string interpolation value, since it's not an expression.")
|
||||||
end
|
end
|
||||||
if #lua.bits > 0 then
|
if #lua.bits > 0 then
|
||||||
lua:append("..")
|
lua:append("..")
|
||||||
@ -857,11 +863,11 @@ do
|
|||||||
local key, value = tree[1], tree[2]
|
local key, value = tree[1], tree[2]
|
||||||
local key_lua = self:compile(key, compile_actions)
|
local key_lua = self:compile(key, compile_actions)
|
||||||
if not (key_lua.is_value) then
|
if not (key_lua.is_value) then
|
||||||
self:compile_error(tree[1].source, "Cannot use:\n%s\nas a dict key, since it's not an expression.")
|
self:compile_error(tree[1], "Can't use this as a dict key, since it's not an expression.")
|
||||||
end
|
end
|
||||||
local value_lua = value and self:compile(value, compile_actions) or LuaCode.Value(key.source, "true")
|
local value_lua = value and self:compile(value, compile_actions) or LuaCode.Value(key.source, "true")
|
||||||
if not (value_lua.is_value) then
|
if not (value_lua.is_value) then
|
||||||
self:compile_error(tree[2].source, "Cannot use:\n%s\nas a dict value, since it's not an expression.")
|
self:compile_error(tree[2], "Can't use this as a dict value, since it's not an expression.")
|
||||||
end
|
end
|
||||||
local key_str = match(tostring(key_lua), [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=])
|
local key_str = match(tostring(key_lua), [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=])
|
||||||
if key_str then
|
if key_str then
|
||||||
@ -874,7 +880,7 @@ do
|
|||||||
elseif "IndexChain" == _exp_0 then
|
elseif "IndexChain" == _exp_0 then
|
||||||
local lua = self:compile(tree[1], compile_actions)
|
local lua = self:compile(tree[1], compile_actions)
|
||||||
if not (lua.is_value) then
|
if not (lua.is_value) then
|
||||||
self:compile_error(tree[1].source, "Cannot index:\n%s\nsince it's not an expression.")
|
self:compile_error(tree[1], "Can't index into this, since it's not an expression.")
|
||||||
end
|
end
|
||||||
local first_char = sub(tostring(lua), 1, 1)
|
local first_char = sub(tostring(lua), 1, 1)
|
||||||
if first_char == "{" or first_char == '"' or first_char == "[" then
|
if first_char == "{" or first_char == '"' or first_char == "[" then
|
||||||
@ -884,7 +890,7 @@ do
|
|||||||
local key = tree[i]
|
local key = tree[i]
|
||||||
local key_lua = self:compile(key, compile_actions)
|
local key_lua = self:compile(key, compile_actions)
|
||||||
if not (key_lua.is_value) then
|
if not (key_lua.is_value) then
|
||||||
self:compile_error(key.source, "Cannot use:\n%s\nas an index, since it's not an expression.")
|
self:compile_error(key, "Can't use this as an index, since it's not an expression.")
|
||||||
end
|
end
|
||||||
local key_lua_str = tostring(key_lua)
|
local key_lua_str = tostring(key_lua)
|
||||||
do
|
do
|
||||||
@ -904,11 +910,11 @@ do
|
|||||||
elseif "Var" == _exp_0 then
|
elseif "Var" == _exp_0 then
|
||||||
return LuaCode.Value(tree.source, (tree[1]):as_lua_id())
|
return LuaCode.Value(tree.source, (tree[1]):as_lua_id())
|
||||||
elseif "FileChunks" == _exp_0 then
|
elseif "FileChunks" == _exp_0 then
|
||||||
return error("Cannot convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks")
|
return error("Can't convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks")
|
||||||
elseif "Comment" == _exp_0 then
|
elseif "Comment" == _exp_0 then
|
||||||
return LuaCode(tree.source, "")
|
return LuaCode(tree.source, "")
|
||||||
elseif "Error" == _exp_0 then
|
elseif "Error" == _exp_0 then
|
||||||
return error("Cannot compile errors")
|
return error("Can't compile errors")
|
||||||
else
|
else
|
||||||
return error("Unknown type: " .. tostring(tree.type))
|
return error("Unknown type: " .. tostring(tree.type))
|
||||||
end
|
end
|
||||||
@ -935,11 +941,11 @@ do
|
|||||||
end
|
end
|
||||||
local _exp_0 = tree.type
|
local _exp_0 = tree.type
|
||||||
if "FileChunks" == _exp_0 then
|
if "FileChunks" == _exp_0 then
|
||||||
return error("Cannot inline a FileChunks")
|
return error("Can't inline a FileChunks")
|
||||||
elseif "Comment" == _exp_0 then
|
elseif "Comment" == _exp_0 then
|
||||||
return NomsuCode(tree.source, "")
|
return NomsuCode(tree.source, "")
|
||||||
elseif "Error" == _exp_0 then
|
elseif "Error" == _exp_0 then
|
||||||
return error("Cannot compile errors")
|
return error("Can't compile errors")
|
||||||
elseif "Action" == _exp_0 then
|
elseif "Action" == _exp_0 then
|
||||||
local nomsu = NomsuCode(tree.source)
|
local nomsu = NomsuCode(tree.source)
|
||||||
if tree.target then
|
if tree.target then
|
||||||
|
@ -164,28 +164,22 @@ with NomsuCompiler
|
|||||||
num_errs = #errs
|
num_errs = #errs
|
||||||
if num_errs > 0
|
if num_errs > 0
|
||||||
err_strings = [pretty_error{
|
err_strings = [pretty_error{
|
||||||
|
title:"Parse error"
|
||||||
error:t.error, hint:t.hint, source:t\get_source_code!
|
error:t.error, hint:t.hint, source:t\get_source_code!
|
||||||
start:t.source.start, stop:t.source.stop
|
start:t.source.start, stop:t.source.stop, filename:t.source.filename
|
||||||
} for i, t in ipairs(errs) when i <= 3]
|
} for i, t in ipairs(errs) when i <= 3]
|
||||||
if num_errs > 3
|
if num_errs > 3
|
||||||
table.insert(err_strings, "\027[31;1m +#{num_errs-#errs} additional errors...\027[0m\n")
|
table.insert(err_strings, "\027[31;1m +#{num_errs-#errs} additional errors...\027[0m\n")
|
||||||
error(table.concat(err_strings, '\n\n'), 0)
|
error(table.concat(err_strings, '\n\n'), 0)
|
||||||
return tree
|
return tree
|
||||||
|
|
||||||
-- TODO: use pretty_error instead of this
|
.compile_error = (tree, err_msg, hint=nil)=>
|
||||||
.compile_error = (source, err_format_string, ...)=>
|
err_str = pretty_error{
|
||||||
err_format_string = err_format_string\gsub("%%[^s]", "%%%1")
|
title: "Compile error"
|
||||||
file = Files.read(source.filename)
|
error:err_msg, hint:hint, source:tree\get_source_code!
|
||||||
line_starts = Files.get_line_starts(file)
|
start:tree.source.start, stop:tree.source.stop, filename:tree.source.filename
|
||||||
line_no = Files.get_line_number(file, source.start)
|
}
|
||||||
line_start = line_starts[line_no]
|
error(err_str, 0)
|
||||||
src = colored.dim(file\sub(line_start, source.start-1))
|
|
||||||
src ..= colored.underscore colored.bright colored.red(file\sub(source.start, source.stop-1))
|
|
||||||
end_of_line = (line_starts[Files.get_line_number(file, source.stop) + 1] or 0) - 1
|
|
||||||
src ..= colored.dim(file\sub(source.stop, end_of_line-1))
|
|
||||||
src = ' '..src\gsub('\n', '\n ')
|
|
||||||
err_msg = err_format_string\format(src, ...)
|
|
||||||
error("#{source.filename}:#{line_no}: "..err_msg, 0)
|
|
||||||
|
|
||||||
add_lua_bits = (val_or_stmt, code, compile_actions)=>
|
add_lua_bits = (val_or_stmt, code, compile_actions)=>
|
||||||
cls = val_or_stmt == "value" and LuaCode.Value or LuaCode
|
cls = val_or_stmt == "value" and LuaCode.Value or LuaCode
|
||||||
@ -199,8 +193,8 @@ with NomsuCompiler
|
|||||||
else
|
else
|
||||||
bit_lua = @compile(bit, compile_actions)
|
bit_lua = @compile(bit, compile_actions)
|
||||||
unless bit_lua.is_value
|
unless bit_lua.is_value
|
||||||
@compile_error bit.source,
|
@compile_error bit,
|
||||||
"Cannot use:\n%s\nas a string interpolation value, since it's not an expression."
|
"Can't use this as a string interpolation value, since it's not an expression."
|
||||||
lua\append bit_lua
|
lua\append bit_lua
|
||||||
return lua
|
return lua
|
||||||
return operate_on_text code
|
return operate_on_text code
|
||||||
@ -223,8 +217,8 @@ with NomsuCompiler
|
|||||||
else
|
else
|
||||||
bit_lua = @compile(bit)
|
bit_lua = @compile(bit)
|
||||||
unless bit_lua.is_value
|
unless bit_lua.is_value
|
||||||
@compile_error bit.source,
|
@compile_error bit,
|
||||||
"Cannot use:\n%s\nas a string interpolation value, since it's not an expression."
|
"Can't use this as a string interpolation value, since it's not an expression."
|
||||||
add_bit_lua(lua, bit_lua)
|
add_bit_lua(lua, bit_lua)
|
||||||
lua\append ")"
|
lua\append ")"
|
||||||
return lua
|
return lua
|
||||||
@ -242,7 +236,8 @@ with NomsuCompiler
|
|||||||
else
|
else
|
||||||
tok_lua = @compile(tok, compile_actions)
|
tok_lua = @compile(tok, compile_actions)
|
||||||
unless tok_lua.is_value
|
unless tok_lua.is_value
|
||||||
@compile_error tok.source, "Non-expression value inside math expression:\n%s"
|
@compile_error tok,
|
||||||
|
"Can't use this as a value in a math expression, since it's not a value."
|
||||||
tok_lua\parenthesize! if tok.type == "Action"
|
tok_lua\parenthesize! if tok.type == "Action"
|
||||||
lua\append tok_lua
|
lua\append tok_lua
|
||||||
lua\append " " if i < #tree
|
lua\append " " if i < #tree
|
||||||
@ -409,8 +404,10 @@ with NomsuCompiler
|
|||||||
-- TODO: use tail call?
|
-- TODO: use tail call?
|
||||||
ret = compile_action(@, tree, unpack(args))
|
ret = compile_action(@, tree, unpack(args))
|
||||||
if not ret
|
if not ret
|
||||||
@compile_error tree.source,
|
info = debug.getinfo(compile_action, "S")
|
||||||
"Compile-time action:\n%s\nfailed to produce any Lua"
|
@compile_error tree,
|
||||||
|
"The compile-time action here (#{stub}) failed to produce any Lua",
|
||||||
|
"Look at the implementation of (#{stub}) in #{info.short_src\sub(1,200)}:#{info.linedefined} and make sure it's returning Lua code."
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
lua = LuaCode.Value(tree.source)
|
lua = LuaCode.Value(tree.source)
|
||||||
@ -427,14 +424,17 @@ with NomsuCompiler
|
|||||||
arg_lua = @compile(tok, compile_actions)
|
arg_lua = @compile(tok, compile_actions)
|
||||||
unless arg_lua.is_value
|
unless arg_lua.is_value
|
||||||
if tok.type == "Block"
|
if tok.type == "Block"
|
||||||
@compile_error tok.source,
|
@compile_error tok,
|
||||||
("Cannot compile action '#{stub}' with a Block as an argument.\n"..
|
"Can't compile action (#{stub}) with a Block as an argument.",
|
||||||
"Maybe there should be a compile-time action with that name that isn't being found?")
|
"Maybe there should be a compile-time action with that name that isn't being found?"
|
||||||
|
|
||||||
|
elseif tok.type == "Action"
|
||||||
|
@compile_error tok,
|
||||||
|
"Can't use this as an argument to (#{stub}), since it's not an expression, it produces: #{repr arg_lua}",
|
||||||
|
"Check the implementation of (#{tok.stub}) to see if it is actually meant to produce an expression."
|
||||||
else
|
else
|
||||||
@compile_error tok.source,
|
@compile_error tok,
|
||||||
"Cannot use:\n%s\nas an argument to '%s', since it's not an expression, it produces: %s",
|
"Can't use this as an argument to (#{stub}), since it's not an expression, it produces: #{repr arg_lua}"
|
||||||
stub, repr arg_lua
|
|
||||||
insert args, arg_lua
|
insert args, arg_lua
|
||||||
lua\concat_append args, ", "
|
lua\concat_append args, ", "
|
||||||
lua\append ")"
|
lua\append ")"
|
||||||
@ -479,8 +479,8 @@ with NomsuCompiler
|
|||||||
unless bit_lua.is_value
|
unless bit_lua.is_value
|
||||||
src = ' '..gsub(tostring(@compile(bit, compile_actions)), '\n','\n ')
|
src = ' '..gsub(tostring(@compile(bit, compile_actions)), '\n','\n ')
|
||||||
line = "#{bit.source.filename}:#{Files.get_line_number(Files.read(bit.source.filename), bit.source.start)}"
|
line = "#{bit.source.filename}:#{Files.get_line_number(Files.read(bit.source.filename), bit.source.start)}"
|
||||||
@compile_error bit.source,
|
@compile_error bit,
|
||||||
"Cannot use:\n%s\nas a string interpolation value, since it's not an expression."
|
"Can't this as a string interpolation value, since it's not an expression."
|
||||||
if #lua.bits > 0 then lua\append ".."
|
if #lua.bits > 0 then lua\append ".."
|
||||||
if bit.type != "Text"
|
if bit.type != "Text"
|
||||||
bit_lua = LuaCode.Value(bit.source, "stringify(",bit_lua,")")
|
bit_lua = LuaCode.Value(bit.source, "stringify(",bit_lua,")")
|
||||||
@ -510,12 +510,12 @@ with NomsuCompiler
|
|||||||
key, value = tree[1], tree[2]
|
key, value = tree[1], tree[2]
|
||||||
key_lua = @compile(key, compile_actions)
|
key_lua = @compile(key, compile_actions)
|
||||||
unless key_lua.is_value
|
unless key_lua.is_value
|
||||||
@compile_error tree[1].source,
|
@compile_error tree[1],
|
||||||
"Cannot use:\n%s\nas a dict key, since it's not an expression."
|
"Can't use this as a dict key, since it's not an expression."
|
||||||
value_lua = value and @compile(value, compile_actions) or LuaCode.Value(key.source, "true")
|
value_lua = value and @compile(value, compile_actions) or LuaCode.Value(key.source, "true")
|
||||||
unless value_lua.is_value
|
unless value_lua.is_value
|
||||||
@compile_error tree[2].source,
|
@compile_error tree[2],
|
||||||
"Cannot use:\n%s\nas a dict value, since it's not an expression."
|
"Can't use this as a dict value, since it's not an expression."
|
||||||
-- TODO: support arbitrary words here, like operators and unicode
|
-- TODO: support arbitrary words here, like operators and unicode
|
||||||
key_str = match(tostring(key_lua), [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=])
|
key_str = match(tostring(key_lua), [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=])
|
||||||
return if key_str
|
return if key_str
|
||||||
@ -531,8 +531,8 @@ with NomsuCompiler
|
|||||||
when "IndexChain"
|
when "IndexChain"
|
||||||
lua = @compile(tree[1], compile_actions)
|
lua = @compile(tree[1], compile_actions)
|
||||||
unless lua.is_value
|
unless lua.is_value
|
||||||
@compile_error tree[1].source,
|
@compile_error tree[1],
|
||||||
"Cannot index:\n%s\nsince it's not an expression."
|
"Can't index into this, since it's not an expression."
|
||||||
first_char = sub(tostring(lua),1,1)
|
first_char = sub(tostring(lua),1,1)
|
||||||
if first_char == "{" or first_char == '"' or first_char == "["
|
if first_char == "{" or first_char == '"' or first_char == "["
|
||||||
lua\parenthesize!
|
lua\parenthesize!
|
||||||
@ -541,8 +541,8 @@ with NomsuCompiler
|
|||||||
key = tree[i]
|
key = tree[i]
|
||||||
key_lua = @compile(key, compile_actions)
|
key_lua = @compile(key, compile_actions)
|
||||||
unless key_lua.is_value
|
unless key_lua.is_value
|
||||||
@compile_error key.source,
|
@compile_error key,
|
||||||
"Cannot use:\n%s\nas an index, since it's not an expression."
|
"Can't use this as an index, since it's not an expression."
|
||||||
key_lua_str = tostring(key_lua)
|
key_lua_str = tostring(key_lua)
|
||||||
if lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
if lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
||||||
lua\append ".#{lua_id}"
|
lua\append ".#{lua_id}"
|
||||||
@ -562,7 +562,7 @@ with NomsuCompiler
|
|||||||
return LuaCode.Value(tree.source, (tree[1])\as_lua_id!)
|
return LuaCode.Value(tree.source, (tree[1])\as_lua_id!)
|
||||||
|
|
||||||
when "FileChunks"
|
when "FileChunks"
|
||||||
error("Cannot convert FileChunks to a single block of lua, since each chunk's "..
|
error("Can't convert FileChunks to a single block of lua, since each chunk's "..
|
||||||
"compilation depends on the earlier chunks")
|
"compilation depends on the earlier chunks")
|
||||||
|
|
||||||
when "Comment"
|
when "Comment"
|
||||||
@ -570,7 +570,7 @@ with NomsuCompiler
|
|||||||
return LuaCode(tree.source, "")
|
return LuaCode(tree.source, "")
|
||||||
|
|
||||||
when "Error"
|
when "Error"
|
||||||
error("Cannot compile errors")
|
error("Can't compile errors")
|
||||||
|
|
||||||
else
|
else
|
||||||
error("Unknown type: #{tree.type}")
|
error("Unknown type: #{tree.type}")
|
||||||
@ -580,14 +580,14 @@ with NomsuCompiler
|
|||||||
@tree_to_inline_nomsu(tree, parenthesize_blocks, check, len + (nomsu and #tostring(nomsu) or 0))
|
@tree_to_inline_nomsu(tree, parenthesize_blocks, check, len + (nomsu and #tostring(nomsu) or 0))
|
||||||
switch tree.type
|
switch tree.type
|
||||||
when "FileChunks"
|
when "FileChunks"
|
||||||
error("Cannot inline a FileChunks")
|
error("Can't inline a FileChunks")
|
||||||
|
|
||||||
when "Comment"
|
when "Comment"
|
||||||
-- TODO: implement?
|
-- TODO: implement?
|
||||||
return NomsuCode(tree.source, "")
|
return NomsuCode(tree.source, "")
|
||||||
|
|
||||||
when "Error"
|
when "Error"
|
||||||
error("Cannot compile errors")
|
error("Can't compile errors")
|
||||||
|
|
||||||
when "Action"
|
when "Action"
|
||||||
nomsu = NomsuCode(tree.source)
|
nomsu = NomsuCode(tree.source)
|
||||||
|
@ -25,7 +25,7 @@ format_error = function(err)
|
|||||||
else
|
else
|
||||||
pointer = (" "):rep(err_linepos + #fmt_str:format(0) - 1) .. "⬆"
|
pointer = (" "):rep(err_linepos + #fmt_str:format(0) - 1) .. "⬆"
|
||||||
end
|
end
|
||||||
local err_msg = "\027[33;41;1mParse error at " .. tostring(err.filename or '???') .. ":" .. tostring(err_linenum) .. "\027[0m"
|
local err_msg = "\027[33;41;1m" .. tostring(err.title or "Error") .. " at " .. tostring(err.filename or '???') .. ":" .. tostring(err_linenum) .. "\027[0m"
|
||||||
for i = err_linenum - context, err_linenum - 1 do
|
for i = err_linenum - context, err_linenum - 1 do
|
||||||
do
|
do
|
||||||
local line = string2.line(err.source, i)
|
local line = string2.line(err.source, i)
|
||||||
@ -61,9 +61,9 @@ format_error = function(err)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
local box_width = 70
|
local box_width = 70
|
||||||
local err_text = "\027[47;31;1m" .. tostring((" " .. err.error):wrap_to_1(box_width):gsub("\n", "\n\027[47;31;1m "))
|
local err_text = "\027[47;31;1m" .. tostring(string2.wrap(" " .. err.error, box_width, 16):gsub("\n", "\n\027[47;31;1m "))
|
||||||
if err.hint then
|
if err.hint then
|
||||||
err_text = err_text .. "\n\027[47;30m" .. tostring((" Suggestion: " .. tostring(err.hint)):wrap_to_1(box_width):gsub("\n", "\n\027[47;30m "))
|
err_text = err_text .. "\n\027[47;30m" .. tostring(string2.wrap(" Suggestion: " .. tostring(err.hint), box_width, 16):gsub("\n", "\n\027[47;30m "))
|
||||||
end
|
end
|
||||||
err_msg = err_msg .. ("\n\027[33;1m " .. box(err_text):gsub("\n", "\n "))
|
err_msg = err_msg .. ("\n\027[33;1m " .. box(err_text):gsub("\n", "\n "))
|
||||||
for i = err_linenum_end + 1, err_linenum_end + context do
|
for i = err_linenum_end + 1, err_linenum_end + context do
|
||||||
|
@ -23,7 +23,7 @@ format_error = (err)->
|
|||||||
(" ")\rep(err_linepos+#fmt_str\format(0)-1).."╚#{("═")\rep(err_size-2)}╝"
|
(" ")\rep(err_linepos+#fmt_str\format(0)-1).."╚#{("═")\rep(err_size-2)}╝"
|
||||||
else
|
else
|
||||||
(" ")\rep(err_linepos+#fmt_str\format(0)-1).."⬆"
|
(" ")\rep(err_linepos+#fmt_str\format(0)-1).."⬆"
|
||||||
err_msg = "\027[33;41;1mParse error at #{err.filename or '???'}:#{err_linenum}\027[0m"
|
err_msg = "\027[33;41;1m#{err.title or "Error"} at #{err.filename or '???'}:#{err_linenum}\027[0m"
|
||||||
for i=err_linenum-context,err_linenum-1
|
for i=err_linenum-context,err_linenum-1
|
||||||
if line = string2.line(err.source, i)
|
if line = string2.line(err.source, i)
|
||||||
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0m#{line}\027[0m"
|
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0m#{line}\027[0m"
|
||||||
@ -47,9 +47,9 @@ format_error = (err)->
|
|||||||
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0;41;30m#{line}\027[0m"
|
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0;41;30m#{line}\027[0m"
|
||||||
|
|
||||||
box_width = 70
|
box_width = 70
|
||||||
err_text = "\027[47;31;1m#{(" "..err.error)\wrap_to_1(box_width)\gsub("\n", "\n\027[47;31;1m ")}"
|
err_text = "\027[47;31;1m#{string2.wrap(" "..err.error, box_width, 16)\gsub("\n", "\n\027[47;31;1m ")}"
|
||||||
if err.hint
|
if err.hint
|
||||||
err_text ..= "\n\027[47;30m#{(" Suggestion: #{err.hint}")\wrap_to_1(box_width)\gsub("\n", "\n\027[47;30m ")}"
|
err_text ..= "\n\027[47;30m#{string2.wrap(" Suggestion: #{err.hint}", box_width, 16)\gsub("\n", "\n\027[47;30m ")}"
|
||||||
err_msg ..= "\n\027[33;1m "..box(err_text)\gsub("\n", "\n ")
|
err_msg ..= "\n\027[33;1m "..box(err_text)\gsub("\n", "\n ")
|
||||||
|
|
||||||
for i=err_linenum_end+1,err_linenum_end+context
|
for i=err_linenum_end+1,err_linenum_end+context
|
||||||
|
Loading…
Reference in New Issue
Block a user