pattern_test.cpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  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/syntax/paren_contents.h"
  7. #include "gmock/gmock.h"
  8. #include "gtest/gtest.h"
  9. #include "llvm/Support/Casting.h"
  10. namespace Carbon {
  11. namespace {
  12. using llvm::cast;
  13. using llvm::isa;
  14. using testing::ElementsAre;
  15. using testing::IsEmpty;
  16. // Matches a TuplePattern::Field named `name` whose `pattern` is an
  17. // `AutoPattern`.
  18. MATCHER_P(AutoFieldNamed, name, "") {
  19. return arg.name == std::string(name) && isa<AutoPattern>(arg.pattern);
  20. }
  21. TEST(PatternTest, EmptyAsPattern) {
  22. ParenContents<Pattern> contents = {.elements = {},
  23. .has_trailing_comma = false};
  24. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  25. EXPECT_EQ(pattern->LineNumber(), 1);
  26. ASSERT_TRUE(isa<TuplePattern>(pattern));
  27. EXPECT_THAT(cast<TuplePattern>(pattern)->Fields(), IsEmpty());
  28. }
  29. TEST(PatternTest, EmptyAsTuplePattern) {
  30. ParenContents<Pattern> contents = {.elements = {},
  31. .has_trailing_comma = false};
  32. const TuplePattern* tuple =
  33. TuplePatternFromParenContents(/*line_num=*/1, contents);
  34. EXPECT_EQ(tuple->LineNumber(), 1);
  35. EXPECT_THAT(tuple->Fields(), IsEmpty());
  36. }
  37. TEST(PatternTest, UnaryNoCommaAsPattern) {
  38. // Equivalent to a code fragment like
  39. // ```
  40. // (
  41. // auto
  42. // )
  43. // ```
  44. ParenContents<Pattern> contents = {
  45. .elements = {{.name = std::nullopt,
  46. .term = new AutoPattern(/*line_num=*/2)}},
  47. .has_trailing_comma = false};
  48. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  49. EXPECT_EQ(pattern->LineNumber(), 2);
  50. ASSERT_TRUE(isa<AutoPattern>(pattern));
  51. }
  52. TEST(PatternTest, UnaryNoCommaAsTuplePattern) {
  53. ParenContents<Pattern> contents = {
  54. .elements = {{.name = std::nullopt,
  55. .term = new AutoPattern(/*line_num=*/2)}},
  56. .has_trailing_comma = false};
  57. const TuplePattern* tuple =
  58. TuplePatternFromParenContents(/*line_num=*/1, contents);
  59. EXPECT_EQ(tuple->LineNumber(), 1);
  60. EXPECT_THAT(tuple->Fields(), ElementsAre(AutoFieldNamed("0")));
  61. }
  62. TEST(PatternTest, UnaryWithCommaAsPattern) {
  63. ParenContents<Pattern> contents = {
  64. .elements = {{.name = std::nullopt,
  65. .term = new AutoPattern(/*line_num=*/2)}},
  66. .has_trailing_comma = true};
  67. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  68. EXPECT_EQ(pattern->LineNumber(), 1);
  69. ASSERT_TRUE(isa<TuplePattern>(pattern));
  70. EXPECT_THAT(cast<TuplePattern>(pattern)->Fields(),
  71. ElementsAre(AutoFieldNamed("0")));
  72. }
  73. TEST(PatternTest, UnaryWithCommaAsTuplePattern) {
  74. ParenContents<Pattern> contents = {
  75. .elements = {{.name = std::nullopt,
  76. .term = new AutoPattern(/*line_num=*/2)}},
  77. .has_trailing_comma = true};
  78. const TuplePattern* tuple =
  79. TuplePatternFromParenContents(/*line_num=*/1, contents);
  80. EXPECT_EQ(tuple->LineNumber(), 1);
  81. EXPECT_THAT(tuple->Fields(), ElementsAre(AutoFieldNamed("0")));
  82. }
  83. TEST(PatternTest, BinaryAsPattern) {
  84. ParenContents<Pattern> contents = {
  85. .elements = {{.name = std::nullopt,
  86. .term = new AutoPattern(/*line_num=*/2)},
  87. {.name = std::nullopt,
  88. .term = new AutoPattern(/*line_num=*/3)}},
  89. .has_trailing_comma = true};
  90. const Pattern* pattern = PatternFromParenContents(/*line_num=*/1, contents);
  91. EXPECT_EQ(pattern->LineNumber(), 1);
  92. ASSERT_TRUE(isa<TuplePattern>(pattern));
  93. EXPECT_THAT(cast<TuplePattern>(pattern)->Fields(),
  94. ElementsAre(AutoFieldNamed("0"), AutoFieldNamed("1")));
  95. }
  96. TEST(PatternTest, BinaryAsTuplePattern) {
  97. ParenContents<Pattern> contents = {
  98. .elements = {{.name = std::nullopt,
  99. .term = new AutoPattern(/*line_num=*/2)},
  100. {.name = std::nullopt,
  101. .term = new AutoPattern(/*line_num=*/3)}},
  102. .has_trailing_comma = true};
  103. const TuplePattern* tuple =
  104. TuplePatternFromParenContents(/*line_num=*/1, contents);
  105. EXPECT_EQ(tuple->LineNumber(), 1);
  106. EXPECT_THAT(tuple->Fields(),
  107. ElementsAre(AutoFieldNamed("0"), AutoFieldNamed("1")));
  108. }
  109. } // namespace
  110. } // namespace Carbon