pattern_test.cpp 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "executable_semantics/ast/pattern.h"
  5. #include "executable_semantics/ast/expression.h"
  6. #include "executable_semantics/ast/paren_contents.h"
  7. #include "executable_semantics/common/arena.h"
  8. #include "gmock/gmock.h"
  9. #include "gtest/gtest.h"
  10. #include "llvm/Support/Casting.h"
  11. namespace Carbon {
  12. namespace {
  13. using llvm::cast;
  14. using llvm::isa;
  15. using testing::ElementsAre;
  16. using testing::IsEmpty;
  17. // Matches a TuplePattern::Field named `name` whose `pattern` is an
  18. // `AutoPattern`.
  19. MATCHER_P(AutoFieldNamed, name, "") {
  20. return arg.name == std::string(name) && isa<AutoPattern>(*arg.pattern);
  21. }
  22. static auto FakeSourceLoc(int line_num) -> SourceLocation {
  23. return SourceLocation("<test>", line_num);
  24. }
  25. class PatternTest : public ::testing::Test {
  26. protected:
  27. Arena arena;
  28. };
  29. TEST_F(PatternTest, EmptyAsPattern) {
  30. ParenContents<Pattern> contents = {.elements = {},
  31. .has_trailing_comma = false};
  32. Ptr<const Pattern> pattern =
  33. PatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  34. EXPECT_EQ(pattern->SourceLoc(), FakeSourceLoc(1));
  35. ASSERT_TRUE(isa<TuplePattern>(*pattern));
  36. EXPECT_THAT(cast<TuplePattern>(*pattern).Fields(), IsEmpty());
  37. }
  38. TEST_F(PatternTest, EmptyAsTuplePattern) {
  39. ParenContents<Pattern> contents = {.elements = {},
  40. .has_trailing_comma = false};
  41. Ptr<const TuplePattern> tuple =
  42. TuplePatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  43. EXPECT_EQ(tuple->SourceLoc(), FakeSourceLoc(1));
  44. EXPECT_THAT(tuple->Fields(), IsEmpty());
  45. }
  46. TEST_F(PatternTest, UnaryNoCommaAsPattern) {
  47. // Equivalent to a code fragment like
  48. // ```
  49. // (
  50. // auto
  51. // )
  52. // ```
  53. ParenContents<Pattern> contents = {
  54. .elements = {{.name = std::nullopt,
  55. .term = arena.New<AutoPattern>(FakeSourceLoc(2))}},
  56. .has_trailing_comma = false};
  57. Ptr<const Pattern> pattern =
  58. PatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  59. EXPECT_EQ(pattern->SourceLoc(), FakeSourceLoc(2));
  60. ASSERT_TRUE(isa<AutoPattern>(*pattern));
  61. }
  62. TEST_F(PatternTest, UnaryNoCommaAsTuplePattern) {
  63. ParenContents<Pattern> contents = {
  64. .elements = {{.name = std::nullopt,
  65. .term = arena.New<AutoPattern>(FakeSourceLoc(2))}},
  66. .has_trailing_comma = false};
  67. Ptr<const TuplePattern> tuple =
  68. TuplePatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  69. EXPECT_EQ(tuple->SourceLoc(), FakeSourceLoc(1));
  70. EXPECT_THAT(tuple->Fields(), ElementsAre(AutoFieldNamed("0")));
  71. }
  72. TEST_F(PatternTest, UnaryWithCommaAsPattern) {
  73. ParenContents<Pattern> contents = {
  74. .elements = {{.name = std::nullopt,
  75. .term = arena.New<AutoPattern>(FakeSourceLoc(2))}},
  76. .has_trailing_comma = true};
  77. Ptr<const Pattern> pattern =
  78. PatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  79. EXPECT_EQ(pattern->SourceLoc(), FakeSourceLoc(1));
  80. ASSERT_TRUE(isa<TuplePattern>(*pattern));
  81. EXPECT_THAT(cast<TuplePattern>(*pattern).Fields(),
  82. ElementsAre(AutoFieldNamed("0")));
  83. }
  84. TEST_F(PatternTest, UnaryWithCommaAsTuplePattern) {
  85. ParenContents<Pattern> contents = {
  86. .elements = {{.name = std::nullopt,
  87. .term = arena.New<AutoPattern>(FakeSourceLoc(2))}},
  88. .has_trailing_comma = true};
  89. Ptr<const TuplePattern> tuple =
  90. TuplePatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  91. EXPECT_EQ(tuple->SourceLoc(), FakeSourceLoc(1));
  92. EXPECT_THAT(tuple->Fields(), ElementsAre(AutoFieldNamed("0")));
  93. }
  94. TEST_F(PatternTest, BinaryAsPattern) {
  95. ParenContents<Pattern> contents = {
  96. .elements = {{.name = std::nullopt,
  97. .term = arena.New<AutoPattern>(FakeSourceLoc(2))},
  98. {.name = std::nullopt,
  99. .term = arena.New<AutoPattern>(FakeSourceLoc(2))}},
  100. .has_trailing_comma = true};
  101. Ptr<const Pattern> pattern =
  102. PatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  103. EXPECT_EQ(pattern->SourceLoc(), FakeSourceLoc(1));
  104. ASSERT_TRUE(isa<TuplePattern>(*pattern));
  105. EXPECT_THAT(cast<TuplePattern>(*pattern).Fields(),
  106. ElementsAre(AutoFieldNamed("0"), AutoFieldNamed("1")));
  107. }
  108. TEST_F(PatternTest, BinaryAsTuplePattern) {
  109. ParenContents<Pattern> contents = {
  110. .elements = {{.name = std::nullopt,
  111. .term = arena.New<AutoPattern>(FakeSourceLoc(2))},
  112. {.name = std::nullopt,
  113. .term = arena.New<AutoPattern>(FakeSourceLoc(2))}},
  114. .has_trailing_comma = true};
  115. Ptr<const TuplePattern> tuple =
  116. TuplePatternFromParenContents(&arena, FakeSourceLoc(1), contents);
  117. EXPECT_EQ(tuple->SourceLoc(), FakeSourceLoc(1));
  118. EXPECT_THAT(tuple->Fields(),
  119. ElementsAre(AutoFieldNamed("0"), AutoFieldNamed("1")));
  120. }
  121. } // namespace
  122. } // namespace Carbon