Merge pull request #39 from robotpy/more-cppheaderparser-tests
More cppheaderparser tests
This commit is contained in:
commit
2c238058d3
@ -60,14 +60,14 @@ def gentest(infile: str, name: str, outfile: str, verbose: bool) -> None:
|
|||||||
|
|
||||||
stmt = nondefault_repr(data)
|
stmt = nondefault_repr(data)
|
||||||
|
|
||||||
content = content.replace("\n", "\n ")
|
content = ("\n" + content.strip()).replace("\n", "\n ")
|
||||||
|
content = "\n".join(l.rstrip() for l in content.splitlines())
|
||||||
|
|
||||||
stmt = inspect.cleandoc(
|
stmt = inspect.cleandoc(
|
||||||
f'''
|
f'''
|
||||||
|
|
||||||
def test_{name}() -> None:
|
def test_{name}() -> None:
|
||||||
content = """
|
content = """{content}
|
||||||
{content}
|
|
||||||
"""
|
"""
|
||||||
data = parse_string(content, cleandoc=True)
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
@ -43,6 +43,7 @@ from .types import (
|
|||||||
Reference,
|
Reference,
|
||||||
TemplateArgument,
|
TemplateArgument,
|
||||||
TemplateDecl,
|
TemplateDecl,
|
||||||
|
TemplateInst,
|
||||||
TemplateNonTypeParam,
|
TemplateNonTypeParam,
|
||||||
TemplateParam,
|
TemplateParam,
|
||||||
TemplateSpecialization,
|
TemplateSpecialization,
|
||||||
@ -406,16 +407,21 @@ class CxxParser:
|
|||||||
|
|
||||||
def _parse_extern(self, tok: LexToken, doxygen: typing.Optional[str]) -> None:
|
def _parse_extern(self, tok: LexToken, doxygen: typing.Optional[str]) -> None:
|
||||||
|
|
||||||
etok = self.lex.token_if("STRING_LITERAL")
|
etok = self.lex.token_if("STRING_LITERAL", "template")
|
||||||
if etok:
|
if etok:
|
||||||
if self.lex.token_if("{"):
|
if etok.type == "STRING_LITERAL":
|
||||||
state = self._push_state(ExternBlockState, etok.value)
|
if self.lex.token_if("{"):
|
||||||
state.location = tok.location
|
state = self._push_state(ExternBlockState, etok.value)
|
||||||
self.visitor.on_extern_block_start(state)
|
state.location = tok.location
|
||||||
return
|
self.visitor.on_extern_block_start(state)
|
||||||
|
return
|
||||||
|
|
||||||
# an extern variable/function with specific linkage
|
# an extern variable/function with specific linkage
|
||||||
self.lex.return_token(etok)
|
self.lex.return_token(etok)
|
||||||
|
else:
|
||||||
|
# must be an extern template instantitation
|
||||||
|
self._parse_template_instantiation(doxygen, True)
|
||||||
|
return
|
||||||
|
|
||||||
self._parse_declarations(tok, doxygen)
|
self._parse_declarations(tok, doxygen)
|
||||||
|
|
||||||
@ -547,6 +553,9 @@ class CxxParser:
|
|||||||
return TemplateDecl(params)
|
return TemplateDecl(params)
|
||||||
|
|
||||||
def _parse_template(self, tok: LexToken, doxygen: typing.Optional[str]) -> None:
|
def _parse_template(self, tok: LexToken, doxygen: typing.Optional[str]) -> None:
|
||||||
|
if not self.lex.token_peek_if("<"):
|
||||||
|
self._parse_template_instantiation(doxygen, False)
|
||||||
|
return
|
||||||
|
|
||||||
template = self._parse_template_decl()
|
template = self._parse_template_decl()
|
||||||
|
|
||||||
@ -624,6 +633,41 @@ class CxxParser:
|
|||||||
|
|
||||||
return TemplateSpecialization(args)
|
return TemplateSpecialization(args)
|
||||||
|
|
||||||
|
def _parse_template_instantiation(
|
||||||
|
self, doxygen: typing.Optional[str], extern: bool
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
explicit-instantiation: [extern] template declaration
|
||||||
|
"""
|
||||||
|
|
||||||
|
# entry is right after template
|
||||||
|
|
||||||
|
tok = self.lex.token_if("class", "struct")
|
||||||
|
if not tok:
|
||||||
|
raise self._parse_error(tok)
|
||||||
|
|
||||||
|
atok = self.lex.token_if_in_set(self._attribute_start_tokens)
|
||||||
|
if atok:
|
||||||
|
self._consume_attribute(atok)
|
||||||
|
|
||||||
|
typename, _ = self._parse_pqname(None)
|
||||||
|
|
||||||
|
# the last segment must have a specialization
|
||||||
|
last_segment = typename.segments[-1]
|
||||||
|
if (
|
||||||
|
not isinstance(last_segment, NameSpecifier)
|
||||||
|
or not last_segment.specialization
|
||||||
|
):
|
||||||
|
raise self._parse_error(
|
||||||
|
None, "expected extern template to have specialization"
|
||||||
|
)
|
||||||
|
|
||||||
|
self._next_token_must_be(";")
|
||||||
|
|
||||||
|
self.visitor.on_template_inst(
|
||||||
|
self.state, TemplateInst(typename, extern, doxygen)
|
||||||
|
)
|
||||||
|
|
||||||
#
|
#
|
||||||
# Attributes
|
# Attributes
|
||||||
#
|
#
|
||||||
|
@ -39,6 +39,7 @@ from .types import (
|
|||||||
FriendDecl,
|
FriendDecl,
|
||||||
Function,
|
Function,
|
||||||
Method,
|
Method,
|
||||||
|
TemplateInst,
|
||||||
Typedef,
|
Typedef,
|
||||||
UsingAlias,
|
UsingAlias,
|
||||||
UsingDecl,
|
UsingDecl,
|
||||||
@ -102,6 +103,9 @@ class NamespaceScope:
|
|||||||
using_ns: typing.List["UsingNamespace"] = field(default_factory=list)
|
using_ns: typing.List["UsingNamespace"] = field(default_factory=list)
|
||||||
using_alias: typing.List[UsingAlias] = field(default_factory=list)
|
using_alias: typing.List[UsingAlias] = field(default_factory=list)
|
||||||
|
|
||||||
|
#: Explicit template instantiations
|
||||||
|
template_insts: typing.List[TemplateInst] = field(default_factory=list)
|
||||||
|
|
||||||
#: Child namespaces
|
#: Child namespaces
|
||||||
namespaces: typing.Dict[str, "NamespaceScope"] = field(default_factory=dict)
|
namespaces: typing.Dict[str, "NamespaceScope"] = field(default_factory=dict)
|
||||||
|
|
||||||
@ -248,6 +252,10 @@ class SimpleCxxVisitor:
|
|||||||
def on_forward_decl(self, state: State, fdecl: ForwardDecl) -> None:
|
def on_forward_decl(self, state: State, fdecl: ForwardDecl) -> None:
|
||||||
self.block.forward_decls.append(fdecl)
|
self.block.forward_decls.append(fdecl)
|
||||||
|
|
||||||
|
def on_template_inst(self, state: State, inst: TemplateInst) -> None:
|
||||||
|
assert isinstance(self.block, NamespaceScope)
|
||||||
|
self.block.template_insts.append(inst)
|
||||||
|
|
||||||
def on_variable(self, state: State, v: Variable) -> None:
|
def on_variable(self, state: State, v: Variable) -> None:
|
||||||
assert isinstance(self.block, NamespaceScope)
|
assert isinstance(self.block, NamespaceScope)
|
||||||
self.block.variables.append(v)
|
self.block.variables.append(v)
|
||||||
|
@ -390,6 +390,23 @@ class TemplateDecl:
|
|||||||
params: typing.List[TemplateParam] = field(default_factory=list)
|
params: typing.List[TemplateParam] = field(default_factory=list)
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class TemplateInst:
|
||||||
|
"""
|
||||||
|
Explicit template instantiation
|
||||||
|
|
||||||
|
.. code-block:: c++
|
||||||
|
|
||||||
|
template class MyClass<1,2>;
|
||||||
|
|
||||||
|
extern template class MyClass<2,3>;
|
||||||
|
"""
|
||||||
|
|
||||||
|
typename: PQName
|
||||||
|
extern: bool
|
||||||
|
doxygen: typing.Optional[str] = None
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class ForwardDecl:
|
class ForwardDecl:
|
||||||
"""
|
"""
|
||||||
|
@ -14,6 +14,7 @@ from .types import (
|
|||||||
FriendDecl,
|
FriendDecl,
|
||||||
Function,
|
Function,
|
||||||
Method,
|
Method,
|
||||||
|
TemplateInst,
|
||||||
Typedef,
|
Typedef,
|
||||||
UsingAlias,
|
UsingAlias,
|
||||||
UsingDecl,
|
UsingDecl,
|
||||||
@ -94,6 +95,11 @@ class CxxVisitor(Protocol):
|
|||||||
Called when a forward declaration is encountered
|
Called when a forward declaration is encountered
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
def on_template_inst(self, state: State, inst: TemplateInst) -> None:
|
||||||
|
"""
|
||||||
|
Called when an explicit template instantiation is encountered
|
||||||
|
"""
|
||||||
|
|
||||||
def on_variable(self, state: State, v: Variable) -> None:
|
def on_variable(self, state: State, v: Variable) -> None:
|
||||||
...
|
...
|
||||||
|
|
||||||
|
@ -2998,3 +2998,217 @@ def test_class_mutable() -> None:
|
|||||||
]
|
]
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_nested_class_access() -> None:
|
||||||
|
content = """
|
||||||
|
class Outer {
|
||||||
|
struct Inner {
|
||||||
|
void fn();
|
||||||
|
};
|
||||||
|
|
||||||
|
void ofn();
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
classes=[
|
||||||
|
ClassScope(
|
||||||
|
class_decl=ClassDecl(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="Outer")], classkey="class"
|
||||||
|
)
|
||||||
|
),
|
||||||
|
classes=[
|
||||||
|
ClassScope(
|
||||||
|
class_decl=ClassDecl(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="Inner")],
|
||||||
|
classkey="struct",
|
||||||
|
),
|
||||||
|
access="private",
|
||||||
|
),
|
||||||
|
methods=[
|
||||||
|
Method(
|
||||||
|
return_type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[FundamentalSpecifier(name="void")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name=PQName(segments=[NameSpecifier(name="fn")]),
|
||||||
|
parameters=[],
|
||||||
|
access="public",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
],
|
||||||
|
methods=[
|
||||||
|
Method(
|
||||||
|
return_type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[FundamentalSpecifier(name="void")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name=PQName(segments=[NameSpecifier(name="ofn")]),
|
||||||
|
parameters=[],
|
||||||
|
access="private",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_class_with_typedef() -> None:
|
||||||
|
content = """
|
||||||
|
template <class SomeType> class A {
|
||||||
|
public:
|
||||||
|
typedef B <SomeType> C;
|
||||||
|
|
||||||
|
A();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
C aCInstance;
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
classes=[
|
||||||
|
ClassScope(
|
||||||
|
class_decl=ClassDecl(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="A")], classkey="class"
|
||||||
|
),
|
||||||
|
template=TemplateDecl(
|
||||||
|
params=[TemplateTypeParam(typekey="class", name="SomeType")]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
fields=[
|
||||||
|
Field(
|
||||||
|
access="protected",
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(segments=[NameSpecifier(name="C")])
|
||||||
|
),
|
||||||
|
name="aCInstance",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
methods=[
|
||||||
|
Method(
|
||||||
|
return_type=None,
|
||||||
|
name=PQName(segments=[NameSpecifier(name="A")]),
|
||||||
|
parameters=[],
|
||||||
|
access="public",
|
||||||
|
constructor=True,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
typedefs=[
|
||||||
|
Typedef(
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="B",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="SomeType"
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name="C",
|
||||||
|
access="public",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_class_ref_qualifiers() -> None:
|
||||||
|
content = """
|
||||||
|
struct X {
|
||||||
|
void fn0();
|
||||||
|
void fn1() &;
|
||||||
|
void fn2() &&;
|
||||||
|
void fn3() && = 0;
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
classes=[
|
||||||
|
ClassScope(
|
||||||
|
class_decl=ClassDecl(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="X")], classkey="struct"
|
||||||
|
)
|
||||||
|
),
|
||||||
|
methods=[
|
||||||
|
Method(
|
||||||
|
return_type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[FundamentalSpecifier(name="void")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name=PQName(segments=[NameSpecifier(name="fn0")]),
|
||||||
|
parameters=[],
|
||||||
|
access="public",
|
||||||
|
),
|
||||||
|
Method(
|
||||||
|
return_type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[FundamentalSpecifier(name="void")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name=PQName(segments=[NameSpecifier(name="fn1")]),
|
||||||
|
parameters=[],
|
||||||
|
access="public",
|
||||||
|
ref_qualifier="&",
|
||||||
|
),
|
||||||
|
Method(
|
||||||
|
return_type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[FundamentalSpecifier(name="void")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name=PQName(segments=[NameSpecifier(name="fn2")]),
|
||||||
|
parameters=[],
|
||||||
|
access="public",
|
||||||
|
ref_qualifier="&&",
|
||||||
|
),
|
||||||
|
Method(
|
||||||
|
return_type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[FundamentalSpecifier(name="void")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name=PQName(segments=[NameSpecifier(name="fn3")]),
|
||||||
|
parameters=[],
|
||||||
|
access="public",
|
||||||
|
ref_qualifier="&&",
|
||||||
|
pure_virtual=True,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
@ -18,6 +18,7 @@ from cxxheaderparser.types import (
|
|||||||
Reference,
|
Reference,
|
||||||
TemplateArgument,
|
TemplateArgument,
|
||||||
TemplateDecl,
|
TemplateDecl,
|
||||||
|
TemplateInst,
|
||||||
TemplateNonTypeParam,
|
TemplateNonTypeParam,
|
||||||
TemplateSpecialization,
|
TemplateSpecialization,
|
||||||
TemplateTypeParam,
|
TemplateTypeParam,
|
||||||
@ -1814,3 +1815,175 @@ def test_template_specialized_fn_typename_template() -> None:
|
|||||||
]
|
]
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_template_instantiation() -> None:
|
||||||
|
content = """
|
||||||
|
template class MyClass<1,2>;
|
||||||
|
template class __attribute__(("something")) MyClass<3,4>;
|
||||||
|
|
||||||
|
namespace foo {
|
||||||
|
template class MyClass<5,6>;
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
template_insts=[
|
||||||
|
TemplateInst(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="MyClass",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="1")])
|
||||||
|
),
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="2")])
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
extern=False,
|
||||||
|
),
|
||||||
|
TemplateInst(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="MyClass",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="3")])
|
||||||
|
),
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="4")])
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
extern=False,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
namespaces={
|
||||||
|
"foo": NamespaceScope(
|
||||||
|
name="foo",
|
||||||
|
template_insts=[
|
||||||
|
TemplateInst(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="MyClass",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="5")])
|
||||||
|
),
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="6")])
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
extern=False,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_extern_template() -> None:
|
||||||
|
content = """
|
||||||
|
extern template class MyClass<1,2>;
|
||||||
|
extern template class __attribute__(("something")) MyClass<3,4>;
|
||||||
|
|
||||||
|
namespace foo {
|
||||||
|
extern template class MyClass<5,6>;
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
template_insts=[
|
||||||
|
TemplateInst(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="MyClass",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="1")])
|
||||||
|
),
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="2")])
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
extern=True,
|
||||||
|
),
|
||||||
|
TemplateInst(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="MyClass",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="3")])
|
||||||
|
),
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="4")])
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
extern=True,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
namespaces={
|
||||||
|
"foo": NamespaceScope(
|
||||||
|
name="foo",
|
||||||
|
template_insts=[
|
||||||
|
TemplateInst(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="MyClass",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="5")])
|
||||||
|
),
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Value(tokens=[Token(value="6")])
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
extern=True,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
)
|
||||||
|
@ -548,3 +548,171 @@ def test_using_many_things() -> None:
|
|||||||
},
|
},
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_using_template_in_class() -> None:
|
||||||
|
content = """
|
||||||
|
class X {
|
||||||
|
template <typename T>
|
||||||
|
using TT = U<T>;
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
classes=[
|
||||||
|
ClassScope(
|
||||||
|
class_decl=ClassDecl(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="X")], classkey="class"
|
||||||
|
)
|
||||||
|
),
|
||||||
|
using_alias=[
|
||||||
|
UsingAlias(
|
||||||
|
alias="TT",
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="U",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="T"
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
template=TemplateDecl(
|
||||||
|
params=[TemplateTypeParam(typekey="typename", name="T")]
|
||||||
|
),
|
||||||
|
access="private",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_using_typename_in_class() -> None:
|
||||||
|
content = """
|
||||||
|
template <class D> class P {
|
||||||
|
using A = typename f::TP<D>::A;
|
||||||
|
public:
|
||||||
|
using State = typename f::TP<D>::S;
|
||||||
|
P(State st);
|
||||||
|
};
|
||||||
|
"""
|
||||||
|
data = parse_string(content, cleandoc=True)
|
||||||
|
|
||||||
|
assert data == ParsedData(
|
||||||
|
namespace=NamespaceScope(
|
||||||
|
classes=[
|
||||||
|
ClassScope(
|
||||||
|
class_decl=ClassDecl(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="P")], classkey="class"
|
||||||
|
),
|
||||||
|
template=TemplateDecl(
|
||||||
|
params=[TemplateTypeParam(typekey="class", name="D")]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
methods=[
|
||||||
|
Method(
|
||||||
|
return_type=None,
|
||||||
|
name=PQName(segments=[NameSpecifier(name="P")]),
|
||||||
|
parameters=[
|
||||||
|
Parameter(
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[NameSpecifier(name="State")]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
name="st",
|
||||||
|
)
|
||||||
|
],
|
||||||
|
access="public",
|
||||||
|
constructor=True,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
using_alias=[
|
||||||
|
UsingAlias(
|
||||||
|
alias="A",
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(name="f"),
|
||||||
|
NameSpecifier(
|
||||||
|
name="TP",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="D"
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
NameSpecifier(name="A"),
|
||||||
|
],
|
||||||
|
has_typename=True,
|
||||||
|
)
|
||||||
|
),
|
||||||
|
access="private",
|
||||||
|
),
|
||||||
|
UsingAlias(
|
||||||
|
alias="State",
|
||||||
|
type=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(name="f"),
|
||||||
|
NameSpecifier(
|
||||||
|
name="TP",
|
||||||
|
specialization=TemplateSpecialization(
|
||||||
|
args=[
|
||||||
|
TemplateArgument(
|
||||||
|
arg=Type(
|
||||||
|
typename=PQName(
|
||||||
|
segments=[
|
||||||
|
NameSpecifier(
|
||||||
|
name="D"
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
NameSpecifier(name="S"),
|
||||||
|
],
|
||||||
|
has_typename=True,
|
||||||
|
)
|
||||||
|
),
|
||||||
|
access="public",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user