1 //===- unittest/Tooling/StencilTest.cpp -----------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "clang/Tooling/Transformer/Stencil.h"
10 #include "clang/AST/ASTTypeTraits.h"
11 #include "clang/AST/Expr.h"
12 #include "clang/ASTMatchers/ASTMatchers.h"
13 #include "clang/Tooling/FixIt.h"
14 #include "clang/Tooling/Tooling.h"
15 #include "llvm/Support/Error.h"
16 #include "llvm/Testing/Support/Error.h"
17 #include "gmock/gmock.h"
18 #include "gtest/gtest.h"
21 using namespace clang
;
22 using namespace transformer
;
23 using namespace ast_matchers
;
27 using ::llvm::HasValue
;
28 using ::llvm::StringError
;
29 using ::testing::AllOf
;
30 using ::testing::HasSubstr
;
31 using MatchResult
= MatchFinder::MatchResult
;
33 // Create a valid translation-unit from a statement.
34 static std::string
wrapSnippet(StringRef ExtraPreface
,
35 StringRef StatementCode
) {
36 constexpr char Preface
[] = R
"cc(
37 namespace N { class C {}; }
38 namespace { class AnonC {}; }
39 struct S { int Field; };
43 T* operator->() const;
47 template<class T> T desugar() { return T(); };
49 return (Preface
+ ExtraPreface
+ "auto stencil_test_snippet = []{" +
54 static DeclarationMatcher
wrapMatcher(const StatementMatcher
&Matcher
) {
55 return varDecl(hasName("stencil_test_snippet"),
56 hasDescendant(compoundStmt(hasAnySubstatement(Matcher
))));
60 // The AST unit from which `result` is built. We bundle it because it backs
61 // the result. Users are not expected to access it.
62 std::unique_ptr
<ASTUnit
> AstUnit
;
63 // The result to use in the test. References `ast_unit`.
67 // Matches `Matcher` against the statement `StatementCode` and returns the
68 // result. Handles putting the statement inside a function and modifying the
69 // matcher correspondingly. `Matcher` should match one of the statements in
70 // `StatementCode` exactly -- that is, produce exactly one match. However,
71 // `StatementCode` may contain other statements not described by `Matcher`.
72 // `ExtraPreface` (optionally) adds extra decls to the TU, before the code.
73 static std::optional
<TestMatch
> matchStmt(StringRef StatementCode
,
74 StatementMatcher Matcher
,
75 StringRef ExtraPreface
= "") {
76 auto AstUnit
= tooling::buildASTFromCodeWithArgs(
77 wrapSnippet(ExtraPreface
, StatementCode
), {"-Wno-unused-value"});
78 if (AstUnit
== nullptr) {
79 ADD_FAILURE() << "AST construction failed";
82 ASTContext
&Context
= AstUnit
->getASTContext();
83 auto Matches
= ast_matchers::match(wrapMatcher(Matcher
), Context
);
84 // We expect a single, exact match for the statement.
85 if (Matches
.size() != 1) {
86 ADD_FAILURE() << "Wrong number of matches: " << Matches
.size();
89 return TestMatch
{std::move(AstUnit
), MatchResult(Matches
[0], &Context
)};
92 class StencilTest
: public ::testing::Test
{
94 // Verifies that the given stencil fails when evaluated on a valid match
95 // result. Binds a statement to "stmt", a (non-member) ctor-initializer to
96 // "init", an expression to "expr" and a (nameless) declaration to "decl".
97 void testError(const Stencil
&Stencil
,
98 ::testing::Matcher
<std::string
> Matcher
) {
99 const std::string Snippet
= R
"cc(
107 auto StmtMatch
= matchStmt(
111 hasDeclaration(decl(hasDescendant(cxxCtorInitializer(
117 ASSERT_TRUE(StmtMatch
);
118 if (auto ResultOrErr
= Stencil
->eval(StmtMatch
->Result
)) {
119 ADD_FAILURE() << "Expected failure but succeeded: " << *ResultOrErr
;
121 auto Err
= llvm::handleErrors(ResultOrErr
.takeError(),
122 [&Matcher
](const StringError
&Err
) {
123 EXPECT_THAT(Err
.getMessage(), Matcher
);
126 ADD_FAILURE() << "Unhandled error: " << llvm::toString(std::move(Err
));
131 // Tests failures caused by references to unbound nodes. `unbound_id` is the
132 // id that will cause the failure.
133 void testUnboundNodeError(const Stencil
&Stencil
, StringRef UnboundId
) {
135 AllOf(HasSubstr(std::string(UnboundId
)), HasSubstr("not bound")));
139 TEST_F(StencilTest
, SingleStatement
) {
140 StringRef
Condition("C"), Then("T"), Else("E");
141 const std::string Snippet
= R
"cc(
147 auto StmtMatch
= matchStmt(
148 Snippet
, ifStmt(hasCondition(expr().bind(Condition
)),
149 hasThen(stmt().bind(Then
)), hasElse(stmt().bind(Else
))));
150 ASSERT_TRUE(StmtMatch
);
151 // Invert the if-then-else.
153 cat("if (!", node(std::string(Condition
)), ") ",
154 statement(std::string(Else
)), " else ", statement(std::string(Then
)));
155 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
),
156 HasValue("if (!true) return 0; else return 1;"));
159 TEST_F(StencilTest
, UnboundNode
) {
160 const std::string Snippet
= R
"cc(
166 auto StmtMatch
= matchStmt(Snippet
, ifStmt(hasCondition(stmt().bind("a1")),
167 hasThen(stmt().bind("a2"))));
168 ASSERT_TRUE(StmtMatch
);
169 auto Stencil
= cat("if(!", node("a1"), ") ", node("UNBOUND"), ";");
170 auto ResultOrErr
= Stencil
->eval(StmtMatch
->Result
);
171 EXPECT_TRUE(llvm::errorToBool(ResultOrErr
.takeError()))
172 << "Expected unbound node, got " << *ResultOrErr
;
175 // Tests that a stencil with a single parameter (`Id`) evaluates to the expected
176 // string, when `Id` is bound to the expression-statement in `Snippet`.
177 void testExpr(StringRef Id
, StringRef Snippet
, const Stencil
&Stencil
,
178 StringRef Expected
) {
179 auto StmtMatch
= matchStmt(Snippet
, expr().bind(Id
));
180 ASSERT_TRUE(StmtMatch
);
181 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
),
182 HasValue(std::string(Expected
)));
185 void testFailure(StringRef Id
, StringRef Snippet
, const Stencil
&Stencil
,
186 testing::Matcher
<std::string
> MessageMatcher
) {
187 auto StmtMatch
= matchStmt(Snippet
, expr().bind(Id
));
188 ASSERT_TRUE(StmtMatch
);
189 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
),
190 Failed
<StringError
>(testing::Property(
191 &StringError::getMessage
, MessageMatcher
)));
194 TEST_F(StencilTest
, SelectionOp
) {
196 testExpr(Id
, "3;", cat(node(std::string(Id
))), "3");
199 TEST_F(StencilTest
, IfBoundOpBound
) {
201 testExpr(Id
, "3;", ifBound(Id
, cat("5"), cat("7")), "5");
204 TEST_F(StencilTest
, IfBoundOpUnbound
) {
206 testExpr(Id
, "3;", ifBound("other", cat("5"), cat("7")), "7");
209 static auto selectMatcher() {
210 // The `anything` matcher is not bound, to test for none of the cases
212 return expr(anyOf(integerLiteral().bind("int"), cxxBoolLiteral().bind("bool"),
213 floatLiteral().bind("float"), anything()));
216 static auto selectStencil() {
220 {"bool", cat("redundant")},
225 TEST_F(StencilTest
, SelectBoundChooseDetectedMatch
) {
226 std::string Input
= "3;";
227 auto StmtMatch
= matchStmt(Input
, selectMatcher());
228 ASSERT_TRUE(StmtMatch
);
229 EXPECT_THAT_EXPECTED(selectStencil()->eval(StmtMatch
->Result
),
230 HasValue(std::string("I")));
233 TEST_F(StencilTest
, SelectBoundChooseFirst
) {
234 std::string Input
= "true;";
235 auto StmtMatch
= matchStmt(Input
, selectMatcher());
236 ASSERT_TRUE(StmtMatch
);
237 EXPECT_THAT_EXPECTED(selectStencil()->eval(StmtMatch
->Result
),
238 HasValue(std::string("B")));
241 TEST_F(StencilTest
, SelectBoundDiesOnExhaustedCases
) {
242 std::string Input
= "\"string\";";
243 auto StmtMatch
= matchStmt(Input
, selectMatcher());
244 ASSERT_TRUE(StmtMatch
);
245 EXPECT_THAT_EXPECTED(
246 selectStencil()->eval(StmtMatch
->Result
),
247 Failed
<StringError
>(testing::Property(
248 &StringError::getMessage
,
249 AllOf(HasSubstr("selectBound failed"), HasSubstr("no default")))));
252 TEST_F(StencilTest
, SelectBoundSucceedsWithDefault
) {
253 std::string Input
= "\"string\";";
254 auto StmtMatch
= matchStmt(Input
, selectMatcher());
255 ASSERT_TRUE(StmtMatch
);
256 auto Stencil
= selectBound({{"int", cat("I")}}, cat("D"));
257 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
),
258 HasValue(std::string("D")));
261 TEST_F(StencilTest
, ExpressionOpNoParens
) {
263 testExpr(Id
, "3;", expression(Id
), "3");
266 // Don't parenthesize a parens expression.
267 TEST_F(StencilTest
, ExpressionOpNoParensParens
) {
269 testExpr(Id
, "(3);", expression(Id
), "(3)");
272 TEST_F(StencilTest
, ExpressionOpBinaryOpParens
) {
274 testExpr(Id
, "3+4;", expression(Id
), "(3+4)");
277 // `expression` shares code with other ops, so we get sufficient coverage of the
278 // error handling code with this test. If that changes in the future, more error
279 // tests should be added.
280 TEST_F(StencilTest
, ExpressionOpUnbound
) {
282 testFailure(Id
, "3;", expression("ACACA"),
283 AllOf(HasSubstr("ACACA"), HasSubstr("not bound")));
286 TEST_F(StencilTest
, DerefPointer
) {
288 testExpr(Id
, "int *x; x;", deref(Id
), "*x");
291 TEST_F(StencilTest
, DerefBinOp
) {
293 testExpr(Id
, "int *x; x + 1;", deref(Id
), "*(x + 1)");
296 TEST_F(StencilTest
, DerefAddressExpr
) {
298 testExpr(Id
, "int x; &x;", deref(Id
), "x");
301 TEST_F(StencilTest
, AddressOfValue
) {
303 testExpr(Id
, "int x; x;", addressOf(Id
), "&x");
306 TEST_F(StencilTest
, AddressOfDerefExpr
) {
308 testExpr(Id
, "int *x; *x;", addressOf(Id
), "x");
311 TEST_F(StencilTest
, MaybeDerefValue
) {
313 testExpr(Id
, "int x; x;", maybeDeref(Id
), "x");
316 TEST_F(StencilTest
, MaybeDerefPointer
) {
318 testExpr(Id
, "int *x; x;", maybeDeref(Id
), "*x");
321 TEST_F(StencilTest
, MaybeDerefBinOp
) {
323 testExpr(Id
, "int *x; x + 1;", maybeDeref(Id
), "*(x + 1)");
326 TEST_F(StencilTest
, MaybeDerefAddressExpr
) {
328 testExpr(Id
, "int x; &x;", maybeDeref(Id
), "x");
331 TEST_F(StencilTest
, MaybeDerefSmartPointer
) {
333 std::string Snippet
= R
"cc(
334 std::unique_ptr<S> x;
337 testExpr(Id
, Snippet
, maybeDeref(Id
), "*x");
340 TEST_F(StencilTest
, MaybeDerefSmartPointerFromMemberExpr
) {
342 std::string Snippet
= "std::unique_ptr<S> x; x->Field;";
344 matchStmt(Snippet
, memberExpr(hasObjectExpression(expr().bind(Id
))));
345 ASSERT_TRUE(StmtMatch
);
346 const Stencil Stencil
= maybeDeref(Id
);
347 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
), HasValue("*x"));
350 TEST_F(StencilTest
, MaybeAddressOfPointer
) {
352 testExpr(Id
, "int *x; x;", maybeAddressOf(Id
), "x");
355 TEST_F(StencilTest
, MaybeAddressOfValue
) {
357 testExpr(Id
, "int x; x;", addressOf(Id
), "&x");
360 TEST_F(StencilTest
, MaybeAddressOfBinOp
) {
362 testExpr(Id
, "int x; x + 1;", maybeAddressOf(Id
), "&(x + 1)");
365 TEST_F(StencilTest
, MaybeAddressOfDerefExpr
) {
367 testExpr(Id
, "int *x; *x;", addressOf(Id
), "x");
370 TEST_F(StencilTest
, MaybeAddressOfSmartPointer
) {
372 testExpr(Id
, "std::unique_ptr<S> x; x;", maybeAddressOf(Id
), "x");
375 TEST_F(StencilTest
, MaybeAddressOfSmartPointerFromMemberCall
) {
377 std::string Snippet
= "std::unique_ptr<S> x; x->Field;";
379 matchStmt(Snippet
, memberExpr(hasObjectExpression(expr().bind(Id
))));
380 ASSERT_TRUE(StmtMatch
);
381 const Stencil Stencil
= maybeAddressOf(Id
);
382 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
), HasValue("x"));
385 TEST_F(StencilTest
, MaybeAddressOfSmartPointerDerefNoCancel
) {
387 testExpr(Id
, "std::unique_ptr<S> x; *x;", maybeAddressOf(Id
), "&*x");
390 TEST_F(StencilTest
, AccessOpValue
) {
391 StringRef Snippet
= R
"cc(
396 testExpr(Id
, Snippet
, access(Id
, "field"), "x.field");
399 TEST_F(StencilTest
, AccessOpValueExplicitText
) {
400 StringRef Snippet
= R
"cc(
405 testExpr(Id
, Snippet
, access(Id
, cat("field")), "x.field");
408 TEST_F(StencilTest
, AccessOpValueAddress
) {
409 StringRef Snippet
= R
"cc(
414 testExpr(Id
, Snippet
, access(Id
, "field"), "x.field");
417 TEST_F(StencilTest
, AccessOpPointer
) {
418 StringRef Snippet
= R
"cc(
423 testExpr(Id
, Snippet
, access(Id
, "field"), "x->field");
426 TEST_F(StencilTest
, AccessOpPointerDereference
) {
427 StringRef Snippet
= R
"cc(
432 testExpr(Id
, Snippet
, access(Id
, "field"), "x->field");
435 TEST_F(StencilTest
, AccessOpSmartPointer
) {
436 StringRef Snippet
= R
"cc(
437 std::unique_ptr<S> x;
441 testExpr(Id
, Snippet
, access(Id
, "field"), "x->field");
444 TEST_F(StencilTest
, AccessOpSmartPointerDereference
) {
445 StringRef Snippet
= R
"cc(
446 std::unique_ptr<S> x;
450 testExpr(Id
, Snippet
, access(Id
, "field"), "x->field");
453 TEST_F(StencilTest
, AccessOpSmartPointerMemberCall
) {
454 StringRef Snippet
= R
"cc(
455 std::unique_ptr<S> x;
460 matchStmt(Snippet
, memberExpr(hasObjectExpression(expr().bind(Id
))));
461 ASSERT_TRUE(StmtMatch
);
462 EXPECT_THAT_EXPECTED(access(Id
, "field")->eval(StmtMatch
->Result
),
463 HasValue("x->field"));
466 TEST_F(StencilTest
, AccessOpExplicitThis
) {
467 using clang::ast_matchers::hasObjectExpression
;
468 using clang::ast_matchers::memberExpr
;
470 // Set up the code so we can bind to a use of this.
471 StringRef Snippet
= R
"cc(
475 int foo() { return this->x; }
478 auto StmtMatch
= matchStmt(
480 traverse(TK_AsIs
, returnStmt(hasReturnValue(ignoringImplicit(memberExpr(
481 hasObjectExpression(expr().bind("obj"))))))));
482 ASSERT_TRUE(StmtMatch
);
483 const Stencil Stencil
= access("obj", "field");
484 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
),
485 HasValue("this->field"));
488 TEST_F(StencilTest
, AccessOpImplicitThis
) {
489 using clang::ast_matchers::hasObjectExpression
;
490 using clang::ast_matchers::memberExpr
;
492 // Set up the code so we can bind to a use of (implicit) this.
493 StringRef Snippet
= R
"cc(
497 int foo() { return x; }
501 matchStmt(Snippet
, returnStmt(hasReturnValue(ignoringImplicit(memberExpr(
502 hasObjectExpression(expr().bind("obj")))))));
503 ASSERT_TRUE(StmtMatch
);
504 const Stencil Stencil
= access("obj", "field");
505 EXPECT_THAT_EXPECTED(Stencil
->eval(StmtMatch
->Result
), HasValue("field"));
508 TEST_F(StencilTest
, DescribeType
) {
509 std::string Snippet
= "int *x; x;";
510 std::string Expected
= "int *";
512 matchStmt(Snippet
, declRefExpr(hasType(qualType().bind("type"))));
513 ASSERT_TRUE(StmtMatch
);
514 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch
->Result
),
515 HasValue(std::string(Expected
)));
518 TEST_F(StencilTest
, DescribeSugaredType
) {
519 std::string Snippet
= "using Ty = int; Ty *x; x;";
520 std::string Expected
= "Ty *";
522 matchStmt(Snippet
, declRefExpr(hasType(qualType().bind("type"))));
523 ASSERT_TRUE(StmtMatch
);
524 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch
->Result
),
525 HasValue(std::string(Expected
)));
528 TEST_F(StencilTest
, DescribeDeclType
) {
529 std::string Snippet
= "S s; s;";
530 std::string Expected
= "S";
532 matchStmt(Snippet
, declRefExpr(hasType(qualType().bind("type"))));
533 ASSERT_TRUE(StmtMatch
);
534 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch
->Result
),
535 HasValue(std::string(Expected
)));
538 TEST_F(StencilTest
, DescribeQualifiedType
) {
539 std::string Snippet
= "N::C c; c;";
540 std::string Expected
= "N::C";
542 matchStmt(Snippet
, declRefExpr(hasType(qualType().bind("type"))));
543 ASSERT_TRUE(StmtMatch
);
544 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch
->Result
),
545 HasValue(std::string(Expected
)));
548 TEST_F(StencilTest
, DescribeUnqualifiedType
) {
549 std::string Snippet
= "using N::C; C c; c;";
550 std::string Expected
= "C";
552 matchStmt(Snippet
, declRefExpr(hasType(qualType().bind("type"))));
553 ASSERT_TRUE(StmtMatch
);
554 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch
->Result
),
555 HasValue(std::string(Expected
)));
558 TEST_F(StencilTest
, DescribeAnonNamespaceType
) {
559 std::string Snippet
= "auto c = desugar<AnonC>(); c;";
560 std::string Expected
= "(anonymous namespace)::AnonC";
562 matchStmt(Snippet
, declRefExpr(hasType(qualType().bind("type"))));
563 ASSERT_TRUE(StmtMatch
);
564 EXPECT_THAT_EXPECTED(describe("type")->eval(StmtMatch
->Result
),
565 HasValue(std::string(Expected
)));
568 TEST_F(StencilTest
, RunOp
) {
570 auto SimpleFn
= [Id
](const MatchResult
&R
) {
571 return std::string(R
.Nodes
.getNodeAs
<Stmt
>(Id
) != nullptr ? "Bound"
574 testExpr(Id
, "3;", run(SimpleFn
), "Bound");
577 TEST_F(StencilTest
, CatOfMacroRangeSucceeds
) {
578 StringRef Snippet
= R
"cpp(
580 double foo(double d);
584 matchStmt(Snippet
, callExpr(callee(functionDecl(hasName("foo"))),
586 hasArgument(0, expr().bind("arg"))));
587 ASSERT_TRUE(StmtMatch
);
588 Stencil S
= cat(node("arg"));
589 EXPECT_THAT_EXPECTED(S
->eval(StmtMatch
->Result
), HasValue("MACRO"));
592 TEST_F(StencilTest
, CatOfMacroArgRangeSucceeds
) {
593 StringRef Snippet
= R
"cpp(
594 #define MACRO(a, b) a + b
598 matchStmt(Snippet
, binaryOperator(hasRHS(expr().bind("rhs"))));
599 ASSERT_TRUE(StmtMatch
);
600 Stencil S
= cat(node("rhs"));
601 EXPECT_THAT_EXPECTED(S
->eval(StmtMatch
->Result
), HasValue("3"));
604 TEST_F(StencilTest
, CatOfMacroArgSubRangeSucceeds
) {
605 StringRef Snippet
= R
"cpp(
606 #define MACRO(a, b) a + b
608 MACRO(2, foo(3));)cpp";
610 auto StmtMatch
= matchStmt(
611 Snippet
, binaryOperator(hasRHS(callExpr(
612 callee(functionDecl(hasName("foo"))), argumentCountIs(1),
613 hasArgument(0, expr().bind("arg"))))));
614 ASSERT_TRUE(StmtMatch
);
615 Stencil S
= cat(node("arg"));
616 EXPECT_THAT_EXPECTED(S
->eval(StmtMatch
->Result
), HasValue("3"));
619 TEST_F(StencilTest
, CatOfInvalidRangeFails
) {
620 StringRef Snippet
= R
"cpp(
622 double foo(double d);
626 matchStmt(Snippet
, callExpr(callee(functionDecl(hasName("foo"))),
628 hasArgument(0, expr().bind("arg"))));
629 ASSERT_TRUE(StmtMatch
);
630 Stencil S
= cat(node("arg"));
631 Expected
<std::string
> Result
= S
->eval(StmtMatch
->Result
);
632 ASSERT_FALSE(Result
);
633 llvm::handleAllErrors(Result
.takeError(), [](const llvm::StringError
&E
) {
634 EXPECT_THAT(E
.getMessage(), AllOf(HasSubstr("selected range"),
635 HasSubstr("macro expansion")));
639 // The `StencilToStringTest` tests verify that the string representation of the
640 // stencil combinator matches (as best possible) the spelling of the
641 // combinator's construction. Exceptions include those combinators that have no
642 // explicit spelling (like raw text) and those supporting non-printable
643 // arguments (like `run`, `selection`).
645 TEST(StencilToStringTest
, RawTextOp
) {
646 auto S
= cat("foo bar baz");
647 StringRef Expected
= R
"("foo bar baz
")";
648 EXPECT_EQ(S
->toString(), Expected
);
651 TEST(StencilToStringTest
, RawTextOpEscaping
) {
652 auto S
= cat("foo \"bar\" baz\\n");
653 StringRef Expected
= R
"("foo
\"bar
\" baz
\\n
")";
654 EXPECT_EQ(S
->toString(), Expected
);
657 TEST(StencilToStringTest
, DescribeOp
) {
658 auto S
= describe("Id");
659 StringRef Expected
= R
"repr(describe("Id
"))repr";
660 EXPECT_EQ(S
->toString(), Expected
);
663 TEST(StencilToStringTest
, DebugPrintNodeOp
) {
664 auto S
= dPrint("Id");
665 StringRef Expected
= R
"repr(dPrint("Id
"))repr";
666 EXPECT_EQ(S
->toString(), Expected
);
669 TEST(StencilToStringTest
, ExpressionOp
) {
670 auto S
= expression("Id");
671 StringRef Expected
= R
"repr(expression("Id
"))repr";
672 EXPECT_EQ(S
->toString(), Expected
);
675 TEST(StencilToStringTest
, DerefOp
) {
676 auto S
= deref("Id");
677 StringRef Expected
= R
"repr(deref("Id
"))repr";
678 EXPECT_EQ(S
->toString(), Expected
);
681 TEST(StencilToStringTest
, AddressOfOp
) {
682 auto S
= addressOf("Id");
683 StringRef Expected
= R
"repr(addressOf("Id
"))repr";
684 EXPECT_EQ(S
->toString(), Expected
);
687 TEST(StencilToStringTest
, SelectionOp
) {
688 auto S1
= cat(node("node1"));
689 EXPECT_EQ(S1
->toString(), "selection(...)");
692 TEST(StencilToStringTest
, AccessOpText
) {
693 auto S
= access("Id", "memberData");
694 StringRef Expected
= R
"repr(access("Id
", "memberData
"))repr";
695 EXPECT_EQ(S
->toString(), Expected
);
698 TEST(StencilToStringTest
, AccessOpSelector
) {
699 auto S
= access("Id", cat(name("otherId")));
700 StringRef Expected
= R
"repr(access("Id
", selection(...)))repr";
701 EXPECT_EQ(S
->toString(), Expected
);
704 TEST(StencilToStringTest
, AccessOpStencil
) {
705 auto S
= access("Id", cat("foo_", "bar"));
706 StringRef Expected
= R
"repr(access("Id
", seq("foo_
", "bar
")))repr";
707 EXPECT_EQ(S
->toString(), Expected
);
710 TEST(StencilToStringTest
, IfBoundOp
) {
711 auto S
= ifBound("Id", cat("trueText"), access("exprId", "memberData"));
713 R
"repr(ifBound("Id
", "trueText
", access("exprId
", "memberData
")))repr";
714 EXPECT_EQ(S
->toString(), Expected
);
717 TEST(StencilToStringTest
, SelectBoundOp
) {
718 auto S
= selectBound({
722 StringRef Expected
= R
"repr(selectBound({{"int", "I
"}, {"float", "F
"}}))repr";
723 EXPECT_EQ(S
->toString(), Expected
);
726 TEST(StencilToStringTest
, SelectBoundOpWithOneCase
) {
727 auto S
= selectBound({{"int", cat("I")}});
728 StringRef Expected
= R
"repr(selectBound({{"int", "I
"}}))repr";
729 EXPECT_EQ(S
->toString(), Expected
);
732 TEST(StencilToStringTest
, SelectBoundOpWithDefault
) {
733 auto S
= selectBound({{"int", cat("I")}, {"float", cat("F")}}, cat("D"));
735 R
"cc(selectBound({{"int", "I
"}, {"float", "F
"}}, "D
"))cc";
736 EXPECT_EQ(S
->toString(), Expected
);
739 TEST(StencilToStringTest
, RunOp
) {
740 auto F1
= [](const MatchResult
&R
) { return "foo"; };
742 EXPECT_EQ(S1
->toString(), "run(...)");
745 TEST(StencilToStringTest
, Sequence
) {
746 auto S
= cat("foo", access("x", "m()"), "bar",
747 ifBound("x", cat("t"), access("e", "f")));
748 StringRef Expected
= R
"repr(seq("foo
", access("x
", "m()"), "bar
", )repr"
749 R
"repr(ifBound("x
", "t
", access("e
", "f
"))))repr";
750 EXPECT_EQ(S
->toString(), Expected
);
753 TEST(StencilToStringTest
, SequenceEmpty
) {
755 StringRef Expected
= "seq()";
756 EXPECT_EQ(S
->toString(), Expected
);
759 TEST(StencilToStringTest
, SequenceSingle
) {
761 StringRef Expected
= "\"foo\"";
762 EXPECT_EQ(S
->toString(), Expected
);
765 TEST(StencilToStringTest
, SequenceFromVector
) {
766 auto S
= catVector({cat("foo"), access("x", "m()"), cat("bar"),
767 ifBound("x", cat("t"), access("e", "f"))});
768 StringRef Expected
= R
"repr(seq("foo
", access("x
", "m()"), "bar
", )repr"
769 R
"repr(ifBound("x
", "t
", access("e
", "f
"))))repr";
770 EXPECT_EQ(S
->toString(), Expected
);