moonscript/moonscript/parse.lua

485 lines
13 KiB
Lua
Raw Normal View History

module("moonscript.parse", package.seeall)
2011-05-16 05:04:38 +00:00
2011-05-29 04:27:31 +00:00
local util = require"moonscript.util"
2011-05-16 05:04:38 +00:00
require"lpeg"
local data = require"moonscript.data"
2011-09-27 15:40:18 +00:00
local types = require"moonscript.types"
2011-05-18 07:45:27 +00:00
2011-09-27 15:40:18 +00:00
local ntype = types.ntype
local dump = util.dump
2011-05-22 07:58:17 +00:00
local trim = util.trim
2011-05-21 22:26:46 +00:00
2011-05-18 07:45:27 +00:00
local Stack = data.Stack
2011-05-16 08:53:44 +00:00
2011-05-17 07:59:58 +00:00
local function count_indent(str)
local sum = 0
for v in str:gmatch("[\t ]") do
if v == ' ' then sum = sum + 1 end
if v == '\t' then sum = sum + 4 end
end
return sum
end
2011-05-16 05:04:38 +00:00
local R, S, V, P = lpeg.R, lpeg.S, lpeg.V, lpeg.P
2011-05-23 05:12:21 +00:00
local C, Ct, Cmt, Cg, Cb, Cc = lpeg.C, lpeg.Ct, lpeg.Cmt, lpeg.Cg, lpeg.Cb, lpeg.Cc
2011-05-16 05:04:38 +00:00
2011-09-12 06:31:56 +00:00
lpeg.setmaxstack(10000)
2011-06-02 03:22:29 +00:00
2011-09-12 06:31:44 +00:00
local White = S" \t\r\n"^0
local _Space = S" \t"^0
2011-09-12 06:31:44 +00:00
local Break = P"\r"^-1 * P"\n"
2011-05-18 07:45:27 +00:00
local Stop = Break + -1
2011-05-17 07:59:58 +00:00
local Indent = C(S"\t "^0) / count_indent
2011-05-16 05:04:38 +00:00
2011-09-12 06:31:44 +00:00
local Comment = P"--" * (1 - S"\r\n")^0 * #Stop
local Space = _Space * Comment^-1
local SomeSpace = S" \t"^1 * Comment^-1
2011-05-22 07:58:17 +00:00
local SpaceBreak = Space * Break
local EmptyLine = SpaceBreak
local AlphaNum = R("az", "AZ", "09", "__")
local _Name = C(R("az", "AZ", "__") * AlphaNum^0)
2011-05-24 06:58:10 +00:00
local Name = Space * _Name
local Num = P"0x" * R("09", "af", "AF")^1 +
R"09"^1 * (P"." * R"09"^1)^-1 * (S"eE" * P"-"^-1 * R"09"^1)^-1
Num = Space * (Num / function(value) return {"number", value} end)
2011-05-16 05:04:38 +00:00
2011-05-22 07:58:17 +00:00
local FactorOp = Space * C(S"+-")
local TermOp = Space * C(S"*/%^")
2011-05-16 05:04:38 +00:00
2011-08-13 02:51:50 +00:00
local Shebang = P"#!" * P(1 - Stop)^0
-- can't have P(false) because it causes preceding patterns not to run
local Cut = P(function() return false end)
-- auto declare Proper variables with lpeg.V
local function wrap_env(fn)
2011-08-15 20:26:32 +00:00
local env = getfenv(fn)
2011-05-16 08:53:44 +00:00
return setfenv(fn, setmetatable({}, {
__index = function(self, name)
local value = env[name]
if value ~= nil then return value end
if name:match"^[A-Z][A-Za-z0-9]*$" then
local v = V(name)
rawset(self, name, v)
return v
end
error("unknown variable referenced: "..name)
end
}))
end
2011-05-21 16:36:26 +00:00
function extract_line(str, start_pos)
str = str:sub(start_pos)
m = str:match"^(.-)\n"
if m then return m end
return str:match"^.-$"
end
local function mark(name)
2011-05-16 08:53:44 +00:00
return function(...)
2011-05-21 08:11:25 +00:00
return {name, ...}
2011-05-16 05:04:38 +00:00
end
end
2011-07-04 16:02:17 +00:00
local function insert_pos(pos, value)
if type(value) == "table" then
value[-1] = pos
end
return value
end
local function pos(patt)
return (lpeg.Cp() * patt) / insert_pos
end
local function got(what)
2011-05-21 16:36:26 +00:00
return Cmt("", function(str, pos, ...)
local cap = {...}
print("++ got "..what, "["..extract_line(str, pos).."]")
2011-05-18 07:45:27 +00:00
return true
2011-05-20 07:10:37 +00:00
end)
2011-05-18 07:45:27 +00:00
end
local function flatten(tbl)
2011-05-16 08:53:44 +00:00
if #tbl == 1 then
return tbl[1]
end
return tbl
end
2011-05-16 05:04:38 +00:00
2011-05-21 08:11:25 +00:00
local function flatten_or_mark(name)
return function(tbl)
if #tbl == 1 then return tbl[1] end
table.insert(tbl, 1, name)
return tbl
end
end
-- makes sure the last item in a chain is an index
local _assignable = { index = true, dot = true, slice = true }
local function check_assignable(str, pos, value)
if ntype(value) == "chain" and _assignable[ntype(value[#value])]
or type(value) == "string"
then
return true, value
end
return false
end
local function sym(chars)
return Space * chars
end
local function symx(chars)
return chars
end
local function simple_string(delim, x)
return C(symx(delim)) * C((P('\\'..delim) +
"\\\\" +
(1 - S('\r\n'..delim)))^0) * sym(delim) / mark"string"
end
local function wrap_func_arg(value)
return {"call", {value}}
end
-- DOCME
local function flatten_func(callee, args)
if #args == 0 then return callee end
args = {"call", args}
if ntype(callee) == "chain" then
-- check for colon stub that needs arguments
if ntype(callee[#callee]) == "colon_stub" then
local stub = callee[#callee]
stub[1] = "colon"
table.insert(stub, args)
else
table.insert(callee, args)
end
return callee
end
return {"chain", callee, args}
end
-- wraps a statement that has a line decorator
local function wrap_decorator(stm, dec)
if not dec then return stm end
local arg = {stm, dec}
if dec[1] == "if" then
local _, cond, fail = unpack(dec)
if fail then fail = {"else", {fail}} end
stm = {"if", cond, {stm}, fail}
elseif dec[1] == "comprehension" then
local _, clauses = unpack(dec)
stm = {"comprehension", stm, clauses}
end
return stm
end
-- wrap if statement if there is a conditional decorator
local function wrap_if(stm, cond)
if cond then
local pass, fail = unpack(cond)
if fail then fail = {"else", {fail}} end
return {"if", cond[2], {stm}, fail}
end
return stm
end
local function check_lua_string(str, pos, right, left)
return #left == #right
end
-- :name in table literal
local function self_assign(name)
return {name, name}
end
local err_msg = "Failed to parse:\n [%d] >> %s (%d)"
2011-05-18 07:45:27 +00:00
local build_grammar = wrap_env(function()
2011-05-18 07:45:27 +00:00
local _indent = Stack(0) -- current indent
2011-05-21 22:26:46 +00:00
local last_pos = 0 -- used to know where to report error
2011-05-18 07:45:27 +00:00
local function check_indent(str, pos, indent)
2011-05-21 16:36:26 +00:00
last_pos = pos
2011-05-18 07:45:27 +00:00
return _indent:top() == indent
end
local function advance_indent(str, pos, indent)
if indent > _indent:top() then
_indent:push(indent)
return true
end
end
local function push_indent(str, pos, indent)
_indent:push(indent)
return true
end
2011-05-18 07:45:27 +00:00
local function pop_indent(str, pos)
if not _indent:pop() then error("unexpected outdent") end
return true
end
local keywords = {}
local function key(chars)
keywords[chars] = true
return Space * chars * -AlphaNum
2011-05-18 07:45:27 +00:00
end
2011-05-17 07:59:58 +00:00
local function op(word)
local patt = Space * C(word)
if word:match("^%w*$") then
keywords[word] = true
patt = patt * -AlphaNum
end
return patt
end
2011-06-19 20:00:27 +00:00
local SimpleName = Name -- for table key
2011-05-20 07:10:37 +00:00
-- make sure name is not a keyword
2011-05-21 22:26:46 +00:00
local Name = Cmt(Name, function(str, pos, name)
2011-05-20 07:10:37 +00:00
if keywords[name] then return false end
return true
end) / trim
local Name = sym"@" * Name / mark"self" + Name + Space * "..." / trim
2011-05-20 07:10:37 +00:00
2011-05-16 08:53:44 +00:00
local g = lpeg.P{
2011-05-20 07:10:37 +00:00
File,
2011-08-13 02:51:50 +00:00
File = Shebang^-1 * (Block + Ct""),
2011-05-21 16:36:26 +00:00
Block = Ct(Line * (Break^1 * Line)^0),
CheckIndent = Cmt(Indent, check_indent), -- validates line is in correct indent
Line = (CheckIndent * Statement + Space * #Stop),
Statement = (Import + While + With + For + ForEach + Return
+ ClassDecl + Export + BreakLoop + Ct(ExpList) / flatten_or_mark"explist" * Space) * ((
-- statement decorators
key"if" * Exp * (key"else" * Exp)^-1 * Space / mark"if" +
CompInner / mark"comprehension"
) * Space)^-1 / wrap_decorator,
2011-05-19 07:32:31 +00:00
Body = Space^-1 * Break * EmptyLine^0 * InBlock + Ct(Statement), -- either a statement, or an indented block
2011-05-19 07:32:31 +00:00
Advance = #Cmt(Indent, advance_indent), -- Advances the indent, gives back whitespace for CheckIndent
PushIndent = Cmt(Indent, push_indent),
PopIndent = Cmt("", pop_indent),
InBlock = Advance * Block * PopIndent,
2011-05-21 16:36:26 +00:00
2011-06-23 06:45:03 +00:00
Import = key"import" * Ct(ImportNameList) * key"from" * Exp / mark"import",
ImportName = (sym"\\" * Ct(Cc"colon_stub" * Name) + Name),
2011-05-24 05:43:40 +00:00
ImportNameList = ImportName * (sym"," * ImportName)^0,
NameList = Name * (sym"," * Name)^0,
BreakLoop = Ct(key"break"/trim),
Return = key"return" * (ExpListLow/mark"explist" + C"") / mark"return",
2011-06-22 02:38:07 +00:00
With = key"with" * Exp * key"do"^-1 * Body / mark"with",
2011-05-23 08:16:49 +00:00
If = key"if" * Exp * key"then"^-1 * Body *
((Break * CheckIndent)^-1 * EmptyLine^0 * key"elseif" * Exp * key"then"^-1 * Body / mark"elseif")^0 *
((Break * CheckIndent)^-1 * EmptyLine^0 * key"else" * Body / mark"else")^-1 / mark"if",
2011-05-23 08:16:49 +00:00
While = key"while" * Exp * key"do"^-1 * Body / mark"while",
2011-05-20 07:10:37 +00:00
2011-06-13 15:42:46 +00:00
For = key"for" * (Name * sym"=" * Ct(Exp * sym"," * Exp * (sym"," * Exp)^-1)) *
key"do"^-1 * Body / mark"for",
ForEach = key"for" * Ct(NameList) * key"in" * (sym"*" * Exp / mark"unpack" + Exp) * key"do"^-1 * Body / mark"foreach",
2011-07-15 06:46:16 +00:00
2011-06-02 03:58:17 +00:00
Comprehension = sym"[" * Exp * CompInner * sym"]" / mark"comprehension",
2011-11-05 17:30:20 +00:00
TblComprehension = sym"{" * Exp * (sym"," * Exp)^-1 * CompInner * sym"}" / mark"tblcomprehension",
CompInner = Ct(CompFor * CompClause^0),
2011-06-02 03:58:17 +00:00
CompFor = key"for" * Ct(NameList) * key"in" * (sym"*" * Exp / mark"unpack" + Exp) / mark"for",
CompClause = CompFor + key"when" * Exp / mark"when",
2011-05-28 06:21:29 +00:00
Assign = Ct(AssignableList) * sym"=" * (Ct(With + If) + Ct(TableBlock + ExpListLow)) / mark"assign",
2011-08-20 17:44:08 +00:00
Update = Assignable * ((sym"..=" + sym"+=" + sym"-=" + sym"*=" + sym"/=" + sym"%=")/trim) * Exp / mark"update",
2011-05-22 05:00:31 +00:00
-- we can ignore precedence for now
OtherOps = op"or" + op"and" + op"<=" + op">=" + op"~=" + op"!=" + op"==" + op".." + op"<" + op">",
2011-06-22 02:38:07 +00:00
Assignable = Cmt(DotChain + Chain, check_assignable) + Name,
2011-05-22 05:00:31 +00:00
AssignableList = Assignable * (sym"," * Assignable)^0,
2011-06-02 03:22:29 +00:00
Exp = Ct(Value * ((OtherOps + FactorOp + TermOp) * Value)^0) / flatten_or_mark"exp",
-- Exp = Ct(Factor * (OtherOps * Factor)^0) / flatten_or_mark"exp",
-- Factor = Ct(Term * (FactorOp * Term)^0) / flatten_or_mark"exp",
-- Term = Ct(Value * (TermOp * Value)^0) / flatten_or_mark"exp",
2011-05-21 22:26:46 +00:00
SimpleValue =
If +
With +
ForEach + For + While +
sym"-" * -SomeSpace * Exp / mark"minus" +
sym"#" * Exp / mark"length" +
key"not" * Exp / mark"not" +
2011-11-05 17:30:20 +00:00
TblComprehension +
TableLit +
Comprehension +
Assign + Update + FunLit + String +
Num,
2011-05-22 07:58:17 +00:00
ChainValue = -- a function call or an object access
((Chain + DotChain + Callable) * Ct(InvokeArgs^-1)) / flatten_func,
2011-07-04 16:02:17 +00:00
Value = pos(
SimpleValue +
Ct(KeyValueList) / mark"table" +
2011-07-04 16:02:17 +00:00
ChainValue),
SliceValue = SimpleValue + ChainValue,
2011-05-23 02:28:25 +00:00
String = Space * DoubleString + Space * SingleString + LuaString,
SingleString = simple_string("'"),
DoubleString = simple_string('"'),
2011-05-22 07:58:17 +00:00
2011-09-12 06:31:44 +00:00
LuaString = Cg(LuaStringOpen, "string_open") * Cb"string_open" * Break^-1 *
2011-05-22 07:58:17 +00:00
C((1 - Cmt(C(LuaStringClose) * Cb"string_open", check_lua_string))^0) *
C(LuaStringClose) / mark"string",
LuaStringOpen = sym"[" * P"="^0 * "[" / trim,
LuaStringClose = "]" * P"="^0 * "]",
2011-05-21 22:26:46 +00:00
Callable = Name + Parens / mark"parens",
2011-05-21 22:26:46 +00:00
Parens = sym"(" * Exp * sym")",
2011-06-21 03:35:05 +00:00
FnArgs = symx"(" * Ct(ExpList^-1) * sym")" + sym"!" * -P"=" * Ct"",
2011-05-24 06:58:10 +00:00
ChainTail = (ChainItem^1 * ColonSuffix^-1 + ColonSuffix),
2011-05-22 05:00:31 +00:00
-- a list of funcalls and indexs on a callable
Chain = Callable * (ChainItem^1 * ColonSuffix^-1 + ColonSuffix) / mark"chain",
2011-05-24 06:58:10 +00:00
2011-06-22 02:38:07 +00:00
-- shorthand dot call for use in with statement
DotChain =
(sym"." * Cc(-1) * (_Name / mark"dot") * ChainTail^-1) / mark"chain" +
2011-06-23 06:45:03 +00:00
(sym"\\" * Cc(-1) * (
(_Name * Invoke / mark"colon") * ChainTail^-1 +
2011-06-22 02:38:07 +00:00
(_Name / mark"colon_stub")
)) / mark"chain",
2011-05-24 06:58:10 +00:00
ChainItem =
Invoke +
2011-06-02 03:22:29 +00:00
Slice +
2011-05-24 06:58:10 +00:00
symx"[" * Exp/mark"index" * sym"]" +
symx"." * _Name/mark"dot" +
ColonCall,
Slice = symx"[" * (SliceValue + Cc(1)) * sym"," * (SliceValue + Cc"") *
(sym"," * SliceValue)^-1 *sym"]" / mark"slice",
2011-06-02 03:22:29 +00:00
2011-06-23 06:45:03 +00:00
ColonCall = symx"\\" * (_Name * Invoke) / mark"colon",
ColonSuffix = symx"\\" * _Name / mark"colon_stub",
2011-05-24 06:58:10 +00:00
Invoke = FnArgs/mark"call" +
SingleString / wrap_func_arg +
DoubleString / wrap_func_arg,
2011-05-20 07:10:37 +00:00
2011-05-23 05:12:21 +00:00
TableValue = KeyValue + Ct(Exp),
2011-05-19 07:32:31 +00:00
TableLit = sym"{" * Ct(
TableValueList^-1 * sym","^-1 *
(SpaceBreak * TableLitLine * (sym","^-1 * SpaceBreak * TableLitLine)^0 * sym","^-1)^-1
) * White * sym"}" / mark"table",
2011-05-23 05:12:21 +00:00
TableValueList = TableValue * (sym"," * TableValue)^0,
TableLitLine = PushIndent * ((TableValueList * PopIndent) + (PopIndent * Cut)) + Space,
2011-05-23 02:28:25 +00:00
-- the unbounded table
TableBlockInner = Ct(KeyValueLine * (SpaceBreak^1 * KeyValueLine)^0),
TableBlock = SpaceBreak^1 * Advance * TableBlockInner * PopIndent / mark"table",
2011-05-23 02:28:25 +00:00
2011-06-16 05:41:17 +00:00
ClassDecl = key"class" * Name * (key"extends" * Exp + C"")^-1 * TableBlock / mark"class",
Export = key"export" * (
Cc"class" * ClassDecl +
op"*" + op"^" +
Ct(NameList) * (sym"=" * Ct(ExpListLow))^-1) / mark"export",
2011-06-14 16:28:28 +00:00
KeyValue = (sym":" * Name) / self_assign + Ct((SimpleName + sym"[" * Exp * sym"]") * symx":" * (Exp + TableBlock)),
2011-05-23 05:12:21 +00:00
KeyValueList = KeyValue * (sym"," * KeyValue)^0,
KeyValueLine = CheckIndent * KeyValueList * sym","^-1,
2011-05-23 02:28:25 +00:00
FnArgsDef = sym"(" * Ct(FnArgDefList^-1) *
(key"using" * Ct(NameList + Space * "nil") + Ct"") *
sym")" + Ct"" * Ct"",
2011-07-16 18:00:39 +00:00
FnArgDefList = FnArgDef * (sym"," * FnArgDef)^0,
FnArgDef = Ct(Name * (sym"=" * Exp)^-1),
2011-07-16 18:00:39 +00:00
FunLit = FnArgsDef *
2011-05-24 07:15:11 +00:00
(sym"->" * Cc"slim" + sym"=>" * Cc"fat") *
(Body + Ct"") / mark"fndef",
2011-05-19 07:32:31 +00:00
2011-05-20 07:10:37 +00:00
NameList = Name * (sym"," * Name)^0,
ExpList = Exp * (sym"," * Exp)^0,
ExpListLow = Exp * ((sym"," + sym";") * Exp)^0,
InvokeArgs = ExpList * (sym"," * SpaceBreak * Advance * ArgBlock)^-1,
ArgBlock = ArgLine * (sym"," * SpaceBreak * ArgLine)^0 * PopIndent,
ArgLine = CheckIndent * ExpList
2011-05-16 08:53:44 +00:00
}
2011-05-18 07:45:27 +00:00
2011-05-17 07:59:58 +00:00
return {
2011-05-20 07:10:37 +00:00
_g = White * g * White * -1,
2011-05-17 07:59:58 +00:00
match = function(self, str, ...)
2011-05-21 16:36:26 +00:00
local pos_to_line = function(pos)
return util.pos_to_line(str, pos)
2011-05-17 07:59:58 +00:00
end
local get_line = function(num)
return util.get_line(str, num)
end
2011-06-14 16:28:28 +00:00
local tree
local args = {...}
local pass, err = assert(pcall(function()
2011-06-14 16:28:28 +00:00
tree = self._g:match(str, unpack(args))
end))
2011-06-14 16:28:28 +00:00
2011-05-17 07:59:58 +00:00
if not tree then
2011-05-21 16:36:26 +00:00
local line_no = pos_to_line(last_pos)
2011-07-16 18:51:40 +00:00
local line_str = get_line(line_no) or ""
return nil, err_msg:format(line_no, trim(line_str), _indent:top())
2011-05-17 07:59:58 +00:00
end
return tree
end
}
2011-05-16 08:53:44 +00:00
end)
-- parse a string
-- returns tree, or nil and error message
function string(str)
local g = build_grammar()
return g:match(str)
end