Change balanced token handling to allow mismatched gt/lt tokens
- These can be used for math, so we just assume the code is doing that
This commit is contained in:
parent
b08d8783d4
commit
8a0568c0f5
@ -230,16 +230,24 @@ class CxxParser:
|
|||||||
if tok.type in self._end_balanced_tokens:
|
if tok.type in self._end_balanced_tokens:
|
||||||
expected = match_stack.pop()
|
expected = match_stack.pop()
|
||||||
if tok.type != expected:
|
if tok.type != expected:
|
||||||
# hack: ambiguous right-shift issues here, really
|
# hack: we only claim to parse correct code, so if this
|
||||||
# should be looking at the context
|
# is less than or greater than, assume that the code is
|
||||||
if tok.type == ">":
|
# doing math and so this unexpected item is correct.
|
||||||
tok = self.lex.token_if(">")
|
#
|
||||||
if tok:
|
# If one of the other items on the stack match, pop back
|
||||||
consumed.append(tok)
|
# to that. Otherwise, ignore it and hope for the best
|
||||||
|
if tok.type != ">" and expected != ">":
|
||||||
|
raise self._parse_error(tok, expected)
|
||||||
|
|
||||||
|
for i, maybe in enumerate(reversed(match_stack)):
|
||||||
|
if tok.type == maybe:
|
||||||
|
for _ in range(i + 1):
|
||||||
|
match_stack.pop()
|
||||||
|
break
|
||||||
|
else:
|
||||||
match_stack.append(expected)
|
match_stack.append(expected)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
raise self._parse_error(tok, expected)
|
|
||||||
if len(match_stack) == 0:
|
if len(match_stack) == 0:
|
||||||
return consumed
|
return consumed
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Note: testcases generated via `python -m cxxheaderparser.gentest`
|
# Note: testcases generated via `python -m cxxheaderparser.gentest`
|
||||||
|
|
||||||
|
from cxxheaderparser.errors import CxxParseError
|
||||||
from cxxheaderparser.types import (
|
from cxxheaderparser.types import (
|
||||||
Array,
|
Array,
|
||||||
ClassDecl,
|
ClassDecl,
|
||||||
@ -21,6 +21,9 @@ from cxxheaderparser.types import (
|
|||||||
)
|
)
|
||||||
from cxxheaderparser.simple import ClassScope, NamespaceScope, ParsedData, parse_string
|
from cxxheaderparser.simple import ClassScope, NamespaceScope, ParsedData, parse_string
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
import re
|
||||||
|
|
||||||
|
|
||||||
def test_var_unixwiz_ridiculous() -> None:
|
def test_var_unixwiz_ridiculous() -> None:
|
||||||
# http://unixwiz.net/techtips/reading-cdecl.html
|
# http://unixwiz.net/techtips/reading-cdecl.html
|
||||||
@ -766,3 +769,73 @@ def test_var_extern() -> None:
|
|||||||
]
|
]
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_balanced_with_gt() -> None:
|
||||||
|
"""Tests _consume_balanced_tokens handling of mismatched gt tokens"""
|
||||||
|
content = """
|
||||||
|
int x = (1 >> 2);
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
variables=[
|
||||||
|
Variable(
|
||||||
|
name=PQName(segments=[NameSpecifier(name="x")]),
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(segments=[FundamentalSpecifier(name="int")])
|
||||||
|
),
|
||||||
|
value=Value(
|
||||||
|
tokens=[
|
||||||
|
Token(value="("),
|
||||||
|
Token(value="1"),
|
||||||
|
Token(value=">"),
|
||||||
|
Token(value=">"),
|
||||||
|
Token(value="2"),
|
||||||
|
Token(value=")"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_balanced_with_lt() -> None:
|
||||||
|
"""Tests _consume_balanced_tokens handling of mismatched lt tokens"""
|
||||||
|
content = """
|
||||||
|
bool z = (i < 4);
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
variables=[
|
||||||
|
Variable(
|
||||||
|
name=PQName(segments=[NameSpecifier(name="z")]),
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(segments=[FundamentalSpecifier(name="bool")])
|
||||||
|
),
|
||||||
|
value=Value(
|
||||||
|
tokens=[
|
||||||
|
Token(value="("),
|
||||||
|
Token(value="i"),
|
||||||
|
Token(value="<"),
|
||||||
|
Token(value="4"),
|
||||||
|
Token(value=")"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_balanced_bad_mismatch() -> None:
|
||||||
|
content = """
|
||||||
|
bool z = (12 ]);
|
||||||
|
"""
|
||||||
|
err = "<str>:1: parse error evaluating ']': unexpected ']', expected ')'"
|
||||||
|
with pytest.raises(CxxParseError, match=re.escape(err)):
|
||||||
|
parse_string(content, cleandoc=True)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user