From 721dcadc8ee0d174c588f2d54b97169cedad36fa Mon Sep 17 00:00:00 2001 From: Jamie Thompson Date: Tue, 24 Oct 2023 15:06:13 +0200 Subject: [PATCH] better explain message for 'pattern expected' --- .../dotty/tools/dotc/reporting/messages.scala | 47 ++++++------- tests/neg/i18750-format.check | 68 +++++++++++++++++++ tests/neg/i18750-format.scala | 4 ++ 3 files changed, 94 insertions(+), 25 deletions(-) create mode 100644 tests/neg/i18750-format.check create mode 100644 tests/neg/i18750-format.scala diff --git a/compiler/src/dotty/tools/dotc/reporting/messages.scala b/compiler/src/dotty/tools/dotc/reporting/messages.scala index 0cf410a99bdd..f4960af18ce8 100644 --- a/compiler/src/dotty/tools/dotc/reporting/messages.scala +++ b/compiler/src/dotty/tools/dotc/reporting/messages.scala @@ -973,66 +973,63 @@ extends SyntaxMsg(IllegalStartOfSimplePatternID) { def msg(using Context) = "pattern expected" def explain(using Context) = { val sipCode = - """def f(x: Int, y: Int) = x match { - | case `y` => ... - |} - """ + """def f(x: Int, y: Int) = x match + | case `y` => ...""".stripMargin val constructorPatternsCode = """case class Person(name: String, age: Int) | - |def test(p: Person) = p match { - | case Person(name, age) => ... - |} - """ - val tupplePatternsCode = - """def swap(tuple: (String, Int)): (Int, String) = tuple match { - | case (text, number) => (number, text) - |} - """ + | def test(p: Person) = p match + | case Person(name, age) => ...""".stripMargin + val tuplePatternsCode = + """def swap(tuple: (String, Int)): (Int, String) = tuple match + | case (text, number) => (number, text)""".stripMargin val patternSequencesCode = - """def getSecondValue(list: List[Int]): Int = list match { - | case List(_, second, x:_*) => second - | case _ => 0 - |}""" + """def getSecondValue(list: List[Int]): Int = list match + | case List(_, second, x*) => second + | case _ => 0""".stripMargin i"""|Simple patterns can be divided into several groups: - |- Variable Patterns: ${hl("case x => ...")}. + |- Variable Patterns: ${hl("case x => ...")} or ${hl("case _ => ...")} | It matches any value, and binds the variable name to that value. | A special case is the wild-card pattern _ which is treated as if it was a fresh | variable on each occurrence. | - |- Typed Patterns: ${hl("case x: Int => ...")} or ${hl("case _: Int => ...")}. + |- Typed Patterns: ${hl("case x: Int => ...")} or ${hl("case _: Int => ...")} | This pattern matches any value matched by the specified type; it binds the variable | name to that value. | - |- Literal Patterns: ${hl("case 123 => ...")} or ${hl("case 'A' => ...")}. + |- Given Patterns: ${hl("case given ExecutionContext => ...")} + | This pattern matches any value matched by the specified type; it binds a ${hl("given")} + | instance with the same type to that value. + | + |- Literal Patterns: ${hl("case 123 => ...")} or ${hl("case 'A' => ...")} | This type of pattern matches any value that is equal to the specified literal. | |- Stable Identifier Patterns: | - | $sipCode + | ${hl(sipCode)} | | the match succeeds only if the x argument and the y argument of f are equal. | |- Constructor Patterns: | - | $constructorPatternsCode + | ${hl(constructorPatternsCode)} | | The pattern binds all object's fields to the variable names (name and age, in this | case). | |- Tuple Patterns: | - | $tupplePatternsCode + | ${hl(tuplePatternsCode)} | | Calling: | - | ${hl("""swap(("Luftballons", 99)""")} + | ${hl("""swap(("Luftballons", 99))""")} | | would give ${hl("""(99, "Luftballons")""")} as a result. | |- Pattern Sequences: | - | $patternSequencesCode + | ${hl(patternSequencesCode)} | | Calling: | diff --git a/tests/neg/i18750-format.check b/tests/neg/i18750-format.check new file mode 100644 index 000000000000..a20cb8eaadf1 --- /dev/null +++ b/tests/neg/i18750-format.check @@ -0,0 +1,68 @@ +-- [E032] Syntax Error: tests/neg/i18750-format.scala:4:7 -------------------------------------------------------------- +4 | case # => () // error + | ^ + | pattern expected + |--------------------------------------------------------------------------------------------------------------------- + | Explanation (enabled by `-explain`) + |- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + | Simple patterns can be divided into several groups: + | - Variable Patterns: case x => ... or case _ => ... + | It matches any value, and binds the variable name to that value. + | A special case is the wild-card pattern _ which is treated as if it was a fresh + | variable on each occurrence. + | + | - Typed Patterns: case x: Int => ... or case _: Int => ... + | This pattern matches any value matched by the specified type; it binds the variable + | name to that value. + | + | - Given Patterns: case given ExecutionContext => ... + | This pattern matches any value matched by the specified type; it binds a given + | instance with the same type to that value. + | + | - Literal Patterns: case 123 => ... or case 'A' => ... + | This type of pattern matches any value that is equal to the specified literal. + | + | - Stable Identifier Patterns: + | + | def f(x: Int, y: Int) = x match + | case `y` => ... + | + | the match succeeds only if the x argument and the y argument of f are equal. + | + | - Constructor Patterns: + | + | case class Person(name: String, age: Int) + | + | def test(p: Person) = p match + | case Person(name, age) => ... + | + | The pattern binds all object's fields to the variable names (name and age, in this + | case). + | + | - Tuple Patterns: + | + | def swap(tuple: (String, Int)): (Int, String) = tuple match + | case (text, number) => (number, text) + | + | Calling: + | + | swap(("Luftballons", 99)) + | + | would give (99, "Luftballons") as a result. + | + | - Pattern Sequences: + | + | def getSecondValue(list: List[Int]): Int = list match + | case List(_, second, x*) => second + | case _ => 0 + | + | Calling: + | + | getSecondValue(List(1, 10, 2)) + | + | would give 10 as a result. + | This pattern is possible because a companion object for the List class has a method + | with the following signature: + | + | def unapplySeq[A](x: List[A]): Some[List[A]] + --------------------------------------------------------------------------------------------------------------------- diff --git a/tests/neg/i18750-format.scala b/tests/neg/i18750-format.scala new file mode 100644 index 000000000000..5d007fcbcb04 --- /dev/null +++ b/tests/neg/i18750-format.scala @@ -0,0 +1,4 @@ +//> using options -explain + +def test = 23 match + case # => () // error