pattern_test.cpp 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  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/common/arena.h"
  7. #include "executable_semantics/syntax/paren_contents.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. TEST(PatternTest, EmptyAsPattern) {
  23. ParenContents<Pattern> contents = {.elements = {},
  24. .has_trailing_comma = false};
  25. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  26. EXPECT_EQ(pattern->LineNumber(), 1);
  27. ASSERT_TRUE(isa<TuplePattern>(pattern));
  28. EXPECT_THAT(cast<TuplePattern>(pattern)->Fields(), IsEmpty());
  29. }
  30. TEST(PatternTest, EmptyAsTuplePattern) {
  31. ParenContents<Pattern> contents = {.elements = {},
  32. .has_trailing_comma = false};
  33. const TuplePattern* tuple =
  34. TuplePatternFromParenContents(/*line_num=*/1, contents);
  35. EXPECT_EQ(tuple->LineNumber(), 1);
  36. EXPECT_THAT(tuple->Fields(), IsEmpty());
  37. }
  38. TEST(PatternTest, UnaryNoCommaAsPattern) {
  39. // Equivalent to a code fragment like
  40. // ```
  41. // (
  42. // auto
  43. // )
  44. // ```
  45. ParenContents<Pattern> contents = {
  46. .elements = {{.name = std::nullopt,
  47. .term = global_arena->New<AutoPattern>(/*line_num=*/2)}},
  48. .has_trailing_comma = false};
  49. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  50. EXPECT_EQ(pattern->LineNumber(), 2);
  51. ASSERT_TRUE(isa<AutoPattern>(pattern));
  52. }
  53. TEST(PatternTest, UnaryNoCommaAsTuplePattern) {
  54. ParenContents<Pattern> contents = {
  55. .elements = {{.name = std::nullopt,
  56. .term = global_arena->New<AutoPattern>(/*line_num=*/2)}},
  57. .has_trailing_comma = false};
  58. const TuplePattern* tuple =
  59. TuplePatternFromParenContents(/*line_num=*/1, contents);
  60. EXPECT_EQ(tuple->LineNumber(), 1);
  61. EXPECT_THAT(tuple->Fields(), ElementsAre(AutoFieldNamed("0")));
  62. }
  63. TEST(PatternTest, UnaryWithCommaAsPattern) {
  64. ParenContents<Pattern> contents = {
  65. .elements = {{.name = std::nullopt,
  66. .term = global_arena->New<AutoPattern>(/*line_num=*/2)}},
  67. .has_trailing_comma = true};
  68. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  69. EXPECT_EQ(pattern->LineNumber(), 1);
  70. ASSERT_TRUE(isa<TuplePattern>(pattern));
  71. EXPECT_THAT(cast<TuplePattern>(pattern)->Fields(),
  72. ElementsAre(AutoFieldNamed("0")));
  73. }
  74. TEST(PatternTest, UnaryWithCommaAsTuplePattern) {
  75. ParenContents<Pattern> contents = {
  76. .elements = {{.name = std::nullopt,
  77. .term = global_arena->New<AutoPattern>(/*line_num=*/2)}},
  78. .has_trailing_comma = true};
  79. const TuplePattern* tuple =
  80. TuplePatternFromParenContents(/*line_num=*/1, contents);
  81. EXPECT_EQ(tuple->LineNumber(), 1);
  82. EXPECT_THAT(tuple->Fields(), ElementsAre(AutoFieldNamed("0")));
  83. }
  84. TEST(PatternTest, BinaryAsPattern) {
  85. ParenContents<Pattern> contents = {
  86. .elements = {{.name = std::nullopt,
  87. .term = global_arena->New<AutoPattern>(/*line_num=*/2)},
  88. {.name = std::nullopt,
  89. .term = global_arena->New<AutoPattern>(/*line_num=*/3)}},
  90. .has_trailing_comma = true};
  91. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  92. EXPECT_EQ(pattern->LineNumber(), 1);
  93. ASSERT_TRUE(isa<TuplePattern>(pattern));
  94. EXPECT_THAT(cast<TuplePattern>(pattern)->Fields(),
  95. ElementsAre(AutoFieldNamed("0"), AutoFieldNamed("1")));
  96. }
  97. TEST(PatternTest, BinaryAsTuplePattern) {
  98. ParenContents<Pattern> contents = {
  99. .elements = {{.name = std::nullopt,
  100. .term = global_arena->New<AutoPattern>(/*line_num=*/2)},
  101. {.name = std::nullopt,
  102. .term = global_arena->New<AutoPattern>(/*line_num=*/3)}},
  103. .has_trailing_comma = true};
  104. const TuplePattern* tuple =
  105. TuplePatternFromParenContents(/*line_num=*/1, contents);
  106. EXPECT_EQ(tuple->LineNumber(), 1);
  107. EXPECT_THAT(tuple->Fields(),
  108. ElementsAre(AutoFieldNamed("0"), AutoFieldNamed("1")));
  109. }
  110. } // namespace
  111. } // namespace Carbon