Skip to content

Commit

Permalink
Add assertions for compile-time errors
Browse files Browse the repository at this point in the history
Fixes #36. Previously, it was not possible to write assertions that a
given program does not compile. Now, users can call the
`compileErrors(String): String` macro which takes a string literal as an
argument and returns the compile-time errors as strings (or
empty string if there is no error)

The macro is implemented for both Scala 2.x and Dotty.
  • Loading branch information
Olafur Pall Geirsson committed Feb 16, 2020
1 parent 52f19ba commit 8e24b75
Show file tree
Hide file tree
Showing 6 changed files with 198 additions and 5 deletions.
41 changes: 38 additions & 3 deletions docs/assertions.md
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@

---
id: assertions
title: Writing assertions
Expand Down Expand Up @@ -135,15 +134,21 @@ Windows/Unix newlines and ANSI color codes. The "=> Obtained" section of
`.stripMargin`.

## `intercept()`
Use `intercept()` when you expect a particular exception to be thrown by the test code (i.e. the test succeeds if the given Exception is thrown)

Use `intercept()` when you expect a particular exception to be thrown by the
test code (i.e. the test succeeds if the given exception is thrown).

```scala mdoc:crash
intercept[java.lang.IllegalArgumentException]{
// code expected to throw exception here
}
```

## `interceptMessage()`
Like intercept() except you can also specify a specific message the given Exception must match.

Like `intercept()` except additionally asserts that the thrown exception has a
specific error message.

```scala mdoc:crash
interceptMessage[java.lang.IllegalArgumentException]("argument type mismatch"){
// code expected to throw exception here
Expand All @@ -163,3 +168,33 @@ Use `clues()` to include optional context why the test failed.
```scala mdoc:crash
fail("test failed", clues(a + b))
```

## `compileErrors()`

Use `compileErrors()` to assert that an example code snippet fails with a
specific compile-time error message.

```scala mdoc
assertNoDiff(
compileErrors("Set(2, 1).sorted"),
"""|error: value sorted is not a member of scala.collection.immutable.Set[Int]
|Set(2, 1).sorted
| ^
|""".stripMargin
)
```

The argument to `compileErrors` must be a string literal. It's not possible to
pass in more complicated expressions such as variables or string interpolators.

```scala mdoc:fail
val code = """val x: String = 2"""
compileErrors(code)
compileErrors(s"/* code */ $code")
```

Inline the `code` variable to fix the compile error.

```scala mdoc
compileErrors("val x: String = 2")
```
43 changes: 43 additions & 0 deletions munit/shared/src/main/scala-2/munit/internal/MacroCompat.scala
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ import munit.Clue
import munit.Location
import scala.language.experimental.macros
import scala.reflect.macros.blackbox.Context
import scala.reflect.macros.TypecheckException
import scala.reflect.macros.ParseException

object MacroCompat {

Expand Down Expand Up @@ -57,4 +59,45 @@ object MacroCompat {
valueType
)
}

trait CompileErrorMacro {
def compileErrors(code: String): String = macro compileErrorsImpl
}

def compileErrorsImpl(c: Context)(code: c.Tree): c.Tree = {
import c.universe._
val toParse: String = code match {
case Literal(Constant(literal: String)) => literal
case _ =>
c.abort(
code.pos,
"cannot compile dynamic expressions, only constant literals.\n" +
"To fix this problem, pass in a string literal in double quotes \"...\""
)
}

def formatError(message: String, pos: scala.reflect.api.Position): String =
new StringBuilder()
.append("error:")
.append(if (message.contains('\n')) "\n" else " ")
.append(message)
.append("\n")
.append(pos.lineContent)
.append("\n")
.append(" " * (pos.column - 1))
.append("^")
.toString()

val message: String =
try {
c.typecheck(c.parse(s"{\n$toParse\n}"))
""
} catch {
case e: ParseException =>
formatError(e.getMessage(), e.pos)
case e: TypecheckException =>
formatError(e.getMessage(), e.pos)
}
Literal(Constant(message))
}
}
15 changes: 15 additions & 0 deletions munit/shared/src/main/scala-3/munit/internal/MacroCompat.scala
Original file line number Diff line number Diff line change
Expand Up @@ -28,4 +28,19 @@ object MacroCompat {
'{ new Clue(${Expr(source)}, $value, ${Expr(valueType)}) }
}

trait CompileErrorMacro {
inline def compileErrors(inline code: String): String = {
val errors = scala.compiletime.testing.typeCheckErrors(code)
errors.map { error =>
val indent = " " * (error.column - 1)
val trimMessage = error.message.linesIterator.map { line =>
if (line.matches(" +")) ""
else line
}.mkString("\n")
val separator = if (error.message.contains('\n')) "\n" else " "
s"error:${separator}${trimMessage}\n${error.lineContent}\n${indent}^"
}.mkString("\n")
}
}

}
3 changes: 2 additions & 1 deletion munit/shared/src/main/scala/munit/Assertions.scala
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,10 @@ import scala.util.control.NonFatal
import scala.collection.mutable
import munit.internal.console.AnsiColors
import org.junit.AssumptionViolatedException
import munit.internal.MacroCompat

object Assertions extends Assertions
trait Assertions {
trait Assertions extends MacroCompat.CompileErrorMacro {

val munitLines = new Lines

Expand Down
99 changes: 99 additions & 0 deletions tests/shared/src/test/scala/munit/TypeCheckSuite.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
package munit

class TypeCheckSuite extends FunSuite {

def check(
options: TestOptions,
obtained: String,
compat: Map[String, String]
)(implicit loc: Location): Unit = {
test(options) {
val split = BuildInfo.scalaVersion.split("\\.")
val binaryVersion = split.take(2).mkString(".")
val majorVersion = split.head match {
case "0" => "3"
case n => n
}
val expected = compat
.get(BuildInfo.scalaVersion)
.orElse(compat.get(binaryVersion))
.orElse(compat.get(majorVersion))
.getOrElse {
compat(BuildInfo.scalaVersion)
}
assertNoDiff(obtained, expected)(loc)
}
}

val msg = "Hello"
check(
"not a member",
compileErrors("msg.foobar"),
Map(
"2" ->
"""|error: value foobar is not a member of String
|msg.foobar
| ^
|""".stripMargin,
"3" ->
"""|error:
|value foobar is not a member of String, but could be made available as an extension method.
|
|The following import might fix the problem:
|
| import munit.Clue.generate
|
|msg.foobar
| ^
|""".stripMargin
)
)

check(
"parse error",
compileErrors("val x: = 2"),
Map(
"2" -> """|error: identifier expected but '=' found.
|val x: = 2
| ^
|""".stripMargin,
"3" ->
// NOTE(olafur): I'm not sure what's going on with the second errors but
// that's what Dotty reports.
"""|error: an identifier expected, but eof found
|val x: = 2
| ^
|error: Declaration of value x not allowed here: only classes can have declared but undefined members
|package munit
| ^
|""".stripMargin
)
)

check(
"type mismatch",
compileErrors("val n: Int = msg"),
Map(
"2" ->
"""|error:
|type mismatch;
| found : String
| required: Int
|val n: Int = msg
| ^
|""".stripMargin,
"3" ->
"""|error:
|Found: (TypeCheckSuite.this.msg : String)
|Required: Int
|
|The following import might make progress towards fixing the problem:
|
| import munit.Clue.generate
|
|val n: Int = msg
| ^
|""".stripMargin
)
)
}
2 changes: 1 addition & 1 deletion website/i18n/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
"tagline": "Scala testing library with actionable errors and extensible APIs",
"docs": {
"assertions": {
"title": "assertions"
"title": "Writing assertions"
},
"filtering": {
"title": "Filtering tests"
Expand Down

0 comments on commit 8e24b75

Please sign in to comment.