Newton_Editor/plugins/snippets/cson/speg/peg.py

196 lines
5.7 KiB
Python
Raw Normal View History

2024-03-20 00:43:27 +00:00
import re
import sys
import six
from .errors import ExpectedExpr, UnexpectedExpr, SemanticFailure, raise_parsing_error
from .position import Position, get_line_at_position
from .rules import eof
class _ParseBacktrackError(BaseException):
pass
class _PegState:
def __init__(self, position):
self.position = position
self.vars = {}
self.committed = False
self.failure_pos = None
self.failures = None
class Node:
def __init__(self, value, start_pos, end_pos):
self.value = value
self.start_pos = start_pos
self.end_pos = end_pos
class Parser:
def __init__(self, text):
self._text = text
def __call__(self, r, *args, **kw):
return self._parse(lambda p: p(r, *args, **kw))
def parse(self, r, *args, **kw):
return self._parse(lambda p: p.consume(r, *args, **kw))
def _parse(self, fn):
p = ParsingState(self._text)
try:
return fn(p)
except _ParseBacktrackError:
assert len(p._states) == 1
assert p._states[0].failure_pos is not None
raise_parsing_error(self._text, p._states[0].failure_pos, p._states[0].failures)
class CallstackEntry:
def __init__(self, position, fn, args, kw):
self.position = position
self.fn = fn
self.args = args
self.kw = kw
def __repr__(self):
return '{} at {}:{}'.format(self.fn.__name__, self.position.line, self.position.col)
class ParsingState(object):
def __init__(self, s):
self._s = s
self._states = [_PegState(Position())]
self._re_cache = {}
self._callstack = []
def __call__(self, r, *args, **kw):
st = self._states[-1]
if r is eof:
if st.position.offset != len(self._s):
self._raise(ExpectedExpr, eof)
return ''
elif isinstance(r, six.string_types):
flags = args[0] if args else 0
compiled = self._re_cache.get((r, flags))
if not compiled:
compiled = re.compile(r, flags)
self._re_cache[r, flags] = compiled
m = compiled.match(self._s[st.position.offset:])
if not m:
self._raise(ExpectedExpr, r)
ms = m.group(0)
st.position = st.position.advanced_by(ms)
return ms
else:
kw.pop('err', None)
self._callstack.append(CallstackEntry(st.position, r, args, kw))
try:
return r(self, *args, **kw)
finally:
self._callstack.pop()
def consume(self, r, *args, **kw):
start_pos = self.position()
value = self(r, *args, **kw)
end_pos = self.position()
return Node(value, start_pos, end_pos)
def position(self):
return self._states[-1].position
def __repr__(self):
line, line_offs = get_line_at_position(self._s, self._states[-1].position)
return '<speg.ParsingState at {!r}>'.format('{}*{}'.format(line[:line_offs], line[line_offs:]))
@staticmethod
def eof(p):
return p(eof)
def error(self, *args, **kw):
if not args:
args = ['semantic error']
self._raise(SemanticFailure, args, kw)
def _raise(self, failure_type, *args):
st = self._states[-1]
if st.failure_pos is None or st.failure_pos <= st.position:
failure = failure_type(*args, callstack=tuple(self._callstack))
if st.failure_pos != st.position:
st.failure_pos = st.position
st.failures = set([failure])
else:
st.failures.add(failure)
raise _ParseBacktrackError()
def get(self, key, default=None):
for state in self._states[::-1]:
if key in state.vars:
return state.vars[key][0]
return default
def set(self, key, value):
self._states[-1].vars[key] = value, False
def set_global(self, key, value):
self._states[-1].vars[key] = value, True
def opt(self, *args, **kw):
with self:
return self(*args, **kw)
def not_(self, r, *args, **kw):
self._states.append(_PegState(self._states[-1].position))
try:
n = self.consume(r)
except _ParseBacktrackError:
consumed = False
else:
consumed = True
finally:
self._states.pop()
if consumed:
self._raise(UnexpectedExpr, n.end_pos, r)
def __enter__(self):
self._states[-1].committed = False
self._states.append(_PegState(self._states[-1].position))
def __exit__(self, type, value, traceback):
if type is None:
self.commit()
else:
cur = self._states[-1]
prev = self._states[-2]
assert cur.failure_pos is not None
if prev.failure_pos is None or prev.failure_pos < cur.failure_pos:
prev.failure_pos = cur.failure_pos
prev.failures = cur.failures
elif prev.failure_pos == cur.failure_pos:
prev.failures.update(cur.failures)
self._states.pop()
return type is _ParseBacktrackError
def commit(self):
cur = self._states[-1]
prev = self._states[-2]
for key in cur.vars:
val, g = cur.vars[key]
if not g:
continue
if key in prev.vars:
prev.vars[key] = val, prev.vars[key][1]
else:
prev.vars[key] = val, True
cur.failure_pos = None
prev.position = cur.position
prev.committed = True
def __nonzero__(self):
return self._states[-1].committed
__bool__ = __nonzero__