All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.scalatest.Assertions.scala Maven / Gradle / Ivy

/*
 * Copyright 2001-2008 Artima, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.scalatest

import exceptions.TestCanceledException
import scala.reflect.Manifest
import Assertions.areEqualComparingArraysStructurally
import org.scalautils.LegacyTripleEquals

/**
 * Trait that contains ScalaTest's basic assertion methods.
 *
 * 

* You can use the assertions provided by this trait in any ScalaTest Suite, because Suite * mixes in this trait. This trait is designed to be used independently of anything else in ScalaTest, though, so you * can mix it into anything. (You can alternatively import the methods defined in this trait. For details, see the documentation * for the Assertions companion object. *

* *

* In any Scala program, you can write assertions by invoking assert and passing in a Boolean expression, * such as: *

* *
 * val left = 2
 * val right = 1
 * assert(left == right)
 * 
* *

* If the passed expression is true, assert will return normally. If false, * Scala's assert will complete abruptly with an AssertionError. This behavior is provided by * the assert method defined in object Predef, whose members are implicitly imported into every * Scala source file. This Assertions traits defines another assert method that hides the * one in Predef. It behaves the same, except that if false is passed it throws * TestFailedException instead of AssertionError. The reason it throws TestFailedException * is because TestFailedException carries information about exactly which item in the stack trace represents * the line of test code that failed, which can help users more quickly find an offending line of code in a failing test. *

* *

* If you pass the previous Boolean expression, left == right to assert in a ScalaTest test, a failure * will be reported, but without reporting the left and right values. You can alternatively encode these values in a String passed as * a second argument to assert, like this: *

* *
 * val left = 2
 * val right = 1
 * assert(left == right, left + " did not equal " + right)
 * 
* *

* Using this form of assert, the failure report will include the left and right values, thereby * helping you debug the problem. However, ScalaTest provides the === operator to make this easier. * You use it like this: *

* *
 * val left = 2
 * val right = 1
 * assert(left === right)
 * 
* *

* Because you use === here instead of ==, the failure report will include the left * and right values. For example, the detail message in the thrown TestFailedException from the assert * shown previously will include, "2 did not equal 1". * From this message you will know that the operand on the left had the value 2, and the operand on the right had the value 1. *

* *

* If you're familiar with JUnit, you would use === * in a ScalaTest Suite where you'd use assertEquals in a JUnit TestCase. * The === operator is made possible by an implicit conversion from Any * to Equalizer. If you're curious to understand the mechanics, see the documentation for * Equalizer and the convertToEqualizer method. *

* *

Expected results

* * Although === provides a natural, readable extension to Scala's assert mechanism, * as the operands become lengthy, the code becomes less readable. In addition, the === comparison * doesn't distinguish between actual and expected values. The operands are just called left and right, * because if one were named expected and the other actual, it would be difficult for people to * remember which was which. To help with these limitations of assertions, Suite includes a method called expectResult that * can be used as an alternative to assert with ===. To use expectResult, you place * the expected value in parentheses after expectResult, followed by curly braces containing code * that should result in the expected value. For example: * *
 * val a = 5
 * val b = 2
 * expectResult(2) {
 *   a - b
 * }
 * 
* *

* In this case, the expected value is 2, and the code being tested is a - b. This expectation will fail, and * the detail message in the TestFailedException will read, "Expected 2, but got 3." *

* *

Forcing failures

* *

* If you just need the test to fail, you can write: *

* *
 * fail()
 * 
* *

* Or, if you want the test to fail with a message, write: *

* *
 * fail("I've got a bad feeling about this")
 * 
* *

Intercepted exceptions

* *

* Sometimes you need to test whether a method throws an expected exception under certain circumstances, such * as when invalid arguments are passed to the method. You can do this in the JUnit 3 style, like this: *

* *
 * val s = "hi"
 * try {
 *   s.charAt(-1)
 *   fail()
 * }
 * catch {
 *   case _: IndexOutOfBoundsException => // Expected, so continue
 * }
 * 
* *

* If charAt throws IndexOutOfBoundsException as expected, control will transfer * to the catch case, which does nothing. If, however, charAt fails to throw an exception, * the next statement, fail(), will be run. The fail method always completes abruptly with * a TestFailedException, thereby signaling a failed test. *

* *

* To make this common use case easier to express and read, ScalaTest provides an intercept * method. You use it like this: *

* *
 * val s = "hi"
 * intercept[IndexOutOfBoundsException] {
 *   s.charAt(-1)
 * }
 * 
* *

* This code behaves much like the previous example. If charAt throws an instance of IndexOutOfBoundsException, * intercept will return that exception. But if charAt completes normally, or throws a different * exception, intercept will complete abruptly with a TestFailedException. intercept returns the * caught exception so that you can inspect it further if you wish, for example, to ensure that data contained inside * the exception has the expected values. *

* *

Assumptions

* *

* Trait Assertions also provides methods that allow you to cancel a test. * You would cancel a test if a resource required by the test was unavailable. For example, if a test * requires an external database to be online, and it isn't, the test could be canceled to indicate * it was unable to run because of the missing database. Such a test assumes a database is * available, and you can use the assume method to indicate this at the beginning of * the test, like this: *

* *
 * assume(database.isAvailable)
 * 
* *

* For each overloaded assert method, trait Assertions provides an * overloaded assume method with an identical signature and behavior, except the * assume methods throw TestCanceledException whereas the * assert methods throw TestFailedException. As with assert, * assume hides a Scala method in Predef that performs a similar * function, but throws AssertionError. And just as you can with assert, * you can optionally provide a clue string, or use === to get a more detailed * error message. Here are some examples: *

* *
 * assume(database.isAvailable, "The database was down again")
 * assume(database.getAllUsers.count === 9)
 * 
* *

Forcing cancelations

* *

* For each overloaded fail method, there's a corresponding cancel method * with an identical signature and behavior, except the cancel methods throw * TestCanceledException whereas the fail methods throw * TestFailedException. Thus if you just need to cancel a test, you can write: *

* *
 * cancel()
 * 
* *

* If you want to cancel the test with a message, just place the message in the parentheses: *

* *
 * cancel("Can't run the test because no internet connection was found")
 * 
* *

Getting a clue

* *

* If you want more information that is provided by default by the methods if this trait, * you can supply a "clue" string in one of several ways. * The extra information (or "clues") you provide will * be included in the detail message of the thrown exception. Both * assert and expect provide a way for a clue to be * included directly, intercept does not. * Here's an example of clues provided directly in assert: *

* *
 * assert(1 + 1 === 3, "this is a clue")
 * 
* *

* and in expect: *

* *
 * expect(3, "this is a clue") { 1 + 1 }
 * 
* *

* The exceptions thrown by the previous two statements will include the clue * string, "this is a clue", in the exception's detail message. * To get the same clue in the detail message of an exception thrown * by a failed intercept call requires using withClue: *

* *
 * withClue("this is a clue") {
 *   intercept[IndexOutOfBoundsException] {
 *     "hi".charAt(-1)
 *   }
 * }
 * 
* *

* The withClue method will only prepend the clue string to the detail * message of exception types that mix in the ModifiableMessage trait. * See the documentation for ModifiableMessage for more information. * If you wish to place a clue string after a block of code, see the documentation for * AppendedClues. *

* * @author Bill Venners */ trait Assertions extends LegacyTripleEquals { /* * * Class used via an implicit conversion to enable any two objects to be compared with * === in assertions in tests. For example: * *
   * assert(a === b)
   * 
* *

* The benefit of using assert(a === b) rather than assert(a == b) is * that a TestFailedException produced by the former will include the values of a and b * in its detail message. * The implicit method that performs the conversion from Any to Equalizer is * convertToEqualizer in trait Assertions. *

* *

* In case you're not familiar with how implicit conversions work in Scala, here's a quick explanation. * The convertToEqualizer method in Assertions is defined as an "implicit" method that takes an * Any, which means you can pass in any object, and it will convert it to an Equalizer. * The Equalizer has === defined. Most objects don't have === defined as a method * on them. Take two Strings, for example: *

* *
   * assert("hello" === "world")
   * 
* *

* Given this code, the Scala compiler looks for an === method on class String, because that's the class of * "hello". String doesn't define ===, so the compiler looks for an implicit conversion from * String to something that does have an === method, and it finds the convertToEqualizer method. It * then rewrites the code to this: *

* *
   * assert(convertToEqualizer("hello").===("world"))
   * 
* *

* So inside a Suite (which mixes in Assertions, === will work on anything. The only * situation in which the implicit conversion wouldn't * happen is on types that have an === method already defined. *

* *

* The primary constructor takes one object, left, whose type is being converted to Equalizer. The left * value may be a null reference, because this is allowed by Scala's == operator. *

* * @param left An object to convert to Equalizer, which represents the left value * of an assertion. * * @author Bill Venners */ /* final class Equalizer(left: Any) { /** * The === operation compares this Equalizer's left value (passed * to the constructor, usually via an implicit conversion) with the passed right value * for equality as determined by the expression left == right. * If true, === returns None. Else, === returns * a Some whose String value indicates the left and right values. * *

* In its typical usage, the Option[String] returned by === will be passed to one of two * of trait Assertion' overloaded assert methods. If None, * which indicates the assertion succeeded, assert will return normally. But if Some is passed, * which indicates the assertion failed, assert will throw a TestFailedException whose detail * message will include the String contained inside the Some, which in turn includes the * left and right values. This TestFailedException is typically embedded in a * Report and passed to a Reporter, which can present the left and right * values to the user. *

*/ def ===(right: Any) = if (areEqualComparingArraysStructurally(left, right)) None else { val (leftee, rightee) = Suite.getObjectsForFailureMessage(left, right) Some(FailureMessages("didNotEqual", leftee, rightee)) } /* def !==(right: Any) = if (left != right) None else { val (leftee, rightee) = Suite.getObjectsForFailureMessage(left, right) Some(FailureMessages("equaled", leftee, rightee)) } */ } */ /** * Assert that a boolean condition is true. * If the condition is true, this method returns normally. * Else, it throws TestFailedException. * * @param condition the boolean condition to assert * @throws TestFailedException if the condition is false. */ def assert(condition: Boolean) { if (!condition) throw newAssertionFailedException(None, None, 4) } private[scalatest] def newAssertionFailedException(optionalMessage: Option[Any], optionalCause: Option[Throwable], stackDepth: Int): Throwable = (optionalMessage, optionalCause) match { case (None, None) => new TestFailedException(stackDepth) case (None, Some(cause)) => new TestFailedException(cause, stackDepth) case (Some(message), None) => new TestFailedException(message.toString, stackDepth) case (Some(message), Some(cause)) => new TestFailedException(message.toString, cause, stackDepth) } private def newTestCanceledException(optionalMessage: Option[Any], optionalCause: Option[Throwable], stackDepth: Int): Throwable = (optionalMessage, optionalCause) match { case (None, None) => new TestCanceledException(stackDepth) case (None, Some(cause)) => new TestCanceledException(cause, stackDepth) case (Some(message), None) => new TestCanceledException(message.toString, stackDepth) case (Some(message), Some(cause)) => new TestCanceledException(message.toString, cause, stackDepth) } /** * Assert that a boolean condition, described in String * message, is true. * If the condition is true, this method returns normally. * Else, it throws TestFailedException with the * String obtained by invoking toString on the * specified clue as the exception's detail message. * * @param condition the boolean condition to assert * @param clue An objects whose toString method returns a message to include in a failure report. * @throws TestFailedException if the condition is false. * @throws NullPointerException if message is null. */ def assert(condition: Boolean, clue: Any) { if (!condition) throw newAssertionFailedException(Some(clue), None, 4) } /** * Assert that an Option[String] is None. * If the condition is None, this method returns normally. * Else, it throws TestFailedException with the String * value of the Some, as well as the * String obtained by invoking toString on the * specified clue, * included in the TestFailedException's detail message. * *

* This form of assert is usually called in conjunction with an * implicit conversion to Equalizer, using a === comparison, as in: *

* *
   * assert(a === b, "extra info reported if assertion fails")
   * 
* *

* For more information on how this mechanism works, see the documentation for * Equalizer. *

* * @param o the Option[String] to assert * @param clue An objects whose toString method returns a message to include in a failure report. * @throws TestFailedException if the Option[String] is Some. * @throws NullPointerException if message is null. */ def assert(o: Option[String], clue: Any) { o match { case Some(s) => throw newAssertionFailedException(Some(clue + "\n" + s), None, 4) case None => } } /** * Assert that an Option[String] is None. * If the condition is None, this method returns normally. * Else, it throws TestFailedException with the String * value of the Some included in the TestFailedException's * detail message. * *

* This form of assert is usually called in conjunction with an * implicit conversion to Equalizer, using a === comparison, as in: *

* *
   * assert(a === b)
   * 
* *

* For more information on how this mechanism works, see the documentation for * Equalizer. *

* * @param o the Option[String] to assert * @throws TestFailedException if the Option[String] is Some. */ def assert(o: Option[String]) { o match { case Some(s) => throw newAssertionFailedException(Some(s), None, 4) case None => } } /** * Assume that a boolean condition is true. * If the condition is true, this method returns normally. * Else, it throws TestCanceledException. * * @param condition the boolean condition to assert * @throws TestCanceledException if the condition is false. */ def assume(condition: Boolean) { if (!condition) throw newTestCanceledException(None, None, 3) } /** * Assume that a boolean condition, described in String * message, is true. * If the condition is true, this method returns normally. * Else, it throws TestCanceledException with the * String obtained by invoking toString on the * specified clue as the exception's detail message. * * @param condition the boolean condition to assume * @param clue An objects whose toString method returns a message to include in a failure report. * @throws TestFailedException if the condition is false. * @throws NullPointerException if message is null. */ def assume(condition: Boolean, clue: Any) { if (!condition) throw newTestCanceledException(Some(clue.toString), None, 3) } /** * Assume that an Option[String] is None. * If the condition is None, this method returns normally. * Else, it throws TestCanceledException with the String * value of the Some, as well as the * String obtained by invoking toString on the * specified clue, * included in the TestCanceledException's detail message. * *

* This form of assume is usually called in conjunction with an * implicit conversion to Equalizer, using a === comparison, as in: *

* *
   * assume(a === b, "extra info reported if assertion fails")
   * 
* *

* For more information on how this mechanism works, see the documentation for * Equalizer. *

* * @param o the Option[String] to assert * @param clue An objects whose toString method returns a message to include in a failure report. * @throws TestCanceledException if the Option[String] is Some. * @throws NullPointerException if message is null. */ def assume(o: Option[String], clue: Any) { o match { case Some(s) => throw newTestCanceledException(Some(clue + "\n" + s), None, 3) case None => } } /** * Assume that an Option[String] is None. * If the condition is None, this method returns normally. * Else, it throws TestCanceledException with the String * value of the Some included in the TestCanceledException's * detail message. * *

* This form of assume is usually called in conjunction with an * implicit conversion to Equalizer, using a === comparison, as in: *

* *
   * assert(a === b)
   * 
* *

* For more information on how this mechanism works, see the documentation for * Equalizer. *

* * @param o the Option[String] to assert * @throws TestFailedException if the Option[String] is Some. */ // def assume(o: Option[String]) = throwIfSome(o, (a: Any) => newTestCanceledException(Some(a.toString), None, 3)) def assume(o: Option[String]) { o match { case Some(s) => throw newTestCanceledException(Some(s), None, 3) case None => } } /* def throwIfSome(o: Option[String], exception: (Any) => Throwable) { o match { case Some(s) => throw exception(s) case None => } } */ /* * * Implicit conversion from Any to Equalizer, used to enable * assertions with === comparisons. * *

* For more information * on this mechanism, see the documentation for Equalizer. *

* *

* Because trait Suite mixes in Assertions, this implicit conversion will always be * available by default in ScalaTest Suites. This is the only implicit conversion that is in scope by default in every * ScalaTest Suite. Other implicit conversions offered by ScalaTest, such as those that support the matchers DSL * or invokePrivate, must be explicitly invited into your test code, either by mixing in a trait or importing the * members of its companion object. The reason ScalaTest requires you to invite in implicit conversions (with the exception of the * implicit conversion for === operator) is because if one of ScalaTest's implicit conversions clashes with an * implicit conversion used in the code you are trying to test, your program won't compile. Thus there is a chance that if you * are ever trying to use a library or test some code that also offers an implicit conversion involving a === operator, * you could run into the problem of a compiler error due to an ambiguous implicit conversion. If that happens, you can turn off * the implicit conversion offered by this convertToEqualizer method simply by overriding the method in your * Suite subclass, but not marking it as implicit: *

* *
   * // In your Suite subclass
   * override def convertToEqualizer(left: Any) = new Equalizer(left)
   * 
* * @param left the object whose type to convert to Equalizer. * @throws NullPointerException if left is null. */ // implicit def convertToEqualizer(left: Any) = new Equalizer(left) /* * Intercept and return an instance of the passed exception class (or an instance of a subclass of the * passed class), which is expected to be thrown by the passed function value. This method invokes the passed * function. If it throws an exception that's an instance of the passed class or one of its * subclasses, this method returns that exception. Else, whether the passed function returns normally * or completes abruptly with a different exception, this method throws TestFailedException * whose detail message includes the String obtained by invoking toString on the passed clue. * *

* Note that the passed Class may represent any type, not just Throwable or one of its subclasses. In * Scala, exceptions can be caught based on traits they implement, so it may at times make sense to pass in a class instance for * a trait. If a class instance is passed for a type that could not possibly be used to catch an exception (such as String, * for example), this method will complete abruptly with a TestFailedException. *

* * @param message An object whose toString method returns a message to include in a failure report. * @param f the function value that should throw the expected exception * @return the intercepted exception, if it is of the expected type * @throws TestFailedException if the passed function does not result in a value equal to the * passed expected value. def intercept[T <: AnyRef](message: Any)(f: => Any)(implicit manifest: Manifest[T]): T = { val clazz = manifest.erasure.asInstanceOf[Class[T]] val messagePrefix = if (message.toString.trim.isEmpty) "" else (message +"\n") val caught = try { f None } catch { case u: Throwable => { if (!clazz.isAssignableFrom(u.getClass)) { val s = Resources("wrongException", clazz.getName, u.getClass.getName) throw newAssertionFailedException(Some(messagePrefix + s), Some(u), 4) } else { Some(u) } } } caught match { case None => val message = messagePrefix + Resources("exceptionExpected", clazz.getName) throw newAssertionFailedException(Some(message), None, 4) case Some(e) => e.asInstanceOf[T] // I know this cast will succeed, becuase iSAssignableFrom succeeded above } } THIS DOESN'T OVERLOAD. I THINK I'LL EITHER NEED TO USE interceptWithMessage OR JUST LEAVE IT OUT. FOR NOW I'LL LEAVE IT OUT. */ /** * Intercept and return an exception that's expected to * be thrown by the passed function value. The thrown exception must be an instance of the * type specified by the type parameter of this method. This method invokes the passed * function. If the function throws an exception that's an instance of the specified type, * this method returns that exception. Else, whether the passed function returns normally * or completes abruptly with a different exception, this method throws TestFailedException. * *

* Note that the type specified as this method's type parameter may represent any subtype of * AnyRef, not just Throwable or one of its subclasses. In * Scala, exceptions can be caught based on traits they implement, so it may at times make sense * to specify a trait that the intercepted exception's class must mix in. If a class instance is * passed for a type that could not possibly be used to catch an exception (such as String, * for example), this method will complete abruptly with a TestFailedException. *

* * @param f the function value that should throw the expected exception * @param manifest an implicit Manifest representing the type of the specified * type parameter. * @return the intercepted exception, if it is of the expected type * @throws TestFailedException if the passed function does not complete abruptly with an exception * that's an instance of the specified type * passed expected value. */ def intercept[T <: AnyRef](f: => Any)(implicit manifest: Manifest[T]): T = { val clazz = manifest.erasure.asInstanceOf[Class[T]] val caught = try { f None } catch { case u: Throwable => { if (!clazz.isAssignableFrom(u.getClass)) { val s = Resources("wrongException", clazz.getName, u.getClass.getName) throw newAssertionFailedException(Some(s), Some(u), 4) } else { Some(u) } } } caught match { case None => val message = Resources("exceptionExpected", clazz.getName) throw newAssertionFailedException(Some(message), None, 4) case Some(e) => e.asInstanceOf[T] // I know this cast will succeed, becuase iSAssignableFrom succeeded above } } /* * Intercept and return an instance of the passed exception class (or an instance of a subclass of the * passed class), which is expected to be thrown by the passed function value. This method invokes the passed * function. If it throws an exception that's an instance of the passed class or one of its * subclasses, this method returns that exception. Else, whether the passed function returns normally * or completes abruptly with a different exception, this method throws TestFailedException. * *

* Note that the passed Class may represent any type, not just Throwable or one of its subclasses. In * Scala, exceptions can be caught based on traits they implement, so it may at times make sense to pass in a class instance for * a trait. If a class instance is passed for a type that could not possibly be used to catch an exception (such as String, * for example), this method will complete abruptly with a TestFailedException. *

* * @param clazz a type to which the expected exception class is assignable, i.e., the exception should be an instance of the type represented by clazz. * @param f the function value that should throw the expected exception * @return the intercepted exception, if * @throws TestFailedException if the passed function does not complete abruptly with an exception that is assignable to the * passed Class. * @throws IllegalArgumentException if the passed clazz is not Throwable or * one of its subclasses. */ /* def intercept[T <: AnyRef](clazz: java.lang.Class[T])(f: => Unit): T = { // intercept(clazz)(f)(manifest) "hi".asInstanceOf[T] } */ /* def intercept[T <: AnyRef](clazz: java.lang.Class[T])(f: => Unit)(implicit manifest: Manifest[T]): T = { intercept(clazz)(f)(manifest) } */ /** * Expect that the value passed as expected equals the value passed as actual. * If the actual equals the expected * (as determined by ==), expectResult returns * normally. Else, if actual is not equal to expected, expectResult throws a * TestFailedException whose detail message includes the expected and actual values, as well as the String * obtained by invoking toString on the passed clue. * * @param expected the expected value * @param clue An object whose toString method returns a message to include in a failure report. * @param actual the actual value, which should equal the passed expected value * @throws TestFailedException if the passed actual value does not equal the passed expected value. */ def expectResult(expected: Any, clue: Any)(actual: Any) { if (actual != expected) { val (act, exp) = Suite.getObjectsForFailureMessage(actual, expected) val s = FailureMessages("expectedButGot", exp, act) throw newAssertionFailedException(Some(clue + "\n" + s), None, 4) } } /** * This expect method has been deprecated; Please use expectResult instead. * *

* To get rid of the deprecation warning, simply replace expect with * expectResult. The name expect will be used for a different purposes in * a future version of ScalaTest. *

*/ @deprecated("This expect method has been deprecated. Please replace all invocations of expect with an identical invocation of expectResult instead.") def expect(expected: Any, clue: Any)(actual: Any) { if (actual != expected) { val (act, exp) = Suite.getObjectsForFailureMessage(actual, expected) val s = FailureMessages("expectedButGot", exp, act) throw newAssertionFailedException(Some(clue + "\n" + s), None, 4) } } /** * Expect that the value passed as expected equals the value passed as actual. * If the actual value equals the expected value * (as determined by ==), expectResult returns * normally. Else, expect throws a * TestFailedException whose detail message includes the expected and actual values. * * @param expected the expected value * @param actual the actual value, which should equal the passed expected value * @throws TestFailedException if the passed actual value does not equal the passed expected value. */ def expectResult(expected: Any)(actual: Any) { if (actual != expected) { val (act, exp) = Suite.getObjectsForFailureMessage(actual, expected) val s = FailureMessages("expectedButGot", exp, act) throw newAssertionFailedException(Some(s), None, 4) } } /** * This expect method has been deprecated; Please use expectResult instead. * *

* To get rid of the deprecation warning, simply replace expect with * expectResult. The name expect will be used for a different purposes in * a future version of ScalaTest. *

*/ @deprecated("This expect method has been deprecated. Please replace all invocations of expect with an identical invocation of expectResult instead.") def expect(expected: Any)(actual: Any) { if (actual != expected) { val (act, exp) = Suite.getObjectsForFailureMessage(actual, expected) val s = FailureMessages("expectedButGot", exp, act) throw newAssertionFailedException(Some(s), None, 4) } } /* * TODO: Delete this if sticking with Nothing instead of Unit as result type of fail. *

* The result type of this and the other overloaded fail methods is * Unit instead of Nothing, because Nothing * is a subtype of all other types. If the result type of fail were * Nothing, a block of code that ends in a call to fail() may * fail to compile if the block being passed as a by-name parameter or function to an * overloaded method. The reason is that the compiler selects which overloaded * method to call based on the static types of the parameters passed. Since * Nothing is an instance of everything, it can often make the overloaded * method selection ambiguous. *

* *

* For a concrete example, the Conductor class * in package org.scalatest.concurrent has two overloaded variants of the * thread method: *

* *
   * def thread[T](fun: => T): Thread
   *
   * def thread[T](name: String)(fun: => T): Thread
   * 
* *

* Given these two overloaded methods, the following code will compile given the result type * of fail is Unit, but would not compile if the result type were * Nothing: *

* *
   * thread { fail() }
   * 
* *

* If the result type of fail were Nothing, the type of the by-name parameter * would be inferred to be Nothing, which is a subtype of both T and * String. Thus the call is ambiguous, because the type matches the first parameter type * of both overloaded thread methods. Unit, by constrast, is not * a subtype of String, so it only matches one overloaded variant and compiles just fine. *

*/ /** * Throws TestFailedException to indicate a test failed. */ def fail() = { throw newAssertionFailedException(None, None, 4) } /** * Throws TestFailedException, with the passed * String message as the exception's detail * message, to indicate a test failed. * * @param message A message describing the failure. * @throws NullPointerException if message is null */ def fail(message: String) = { if (message == null) throw new NullPointerException("message is null") throw newAssertionFailedException(Some(message), None, 4) } /** * Throws TestFailedException, with the passed * String message as the exception's detail * message and Throwable cause, to indicate a test failed. * * @param message A message describing the failure. * @param cause A Throwable that indicates the cause of the failure. * @throws NullPointerException if message or cause is null */ def fail(message: String, cause: Throwable) = { if (message == null) throw new NullPointerException("message is null") if (cause == null) throw new NullPointerException("cause is null") throw newAssertionFailedException(Some(message), Some(cause), 4) } /** * Throws TestFailedException, with the passed * Throwable cause, to indicate a test failed. * The getMessage method of the thrown TestFailedException * will return cause.toString. * * @param cause a Throwable that indicates the cause of the failure. * @throws NullPointerException if cause is null */ def fail(cause: Throwable) = { if (cause == null) throw new NullPointerException("cause is null") throw newAssertionFailedException(None, Some(cause), 4) } /** * Throws TestCanceledException to indicate a test was canceled. */ def cancel() = { throw newTestCanceledException(None, None, 3) } /** * Throws TestCanceledException, with the passed * String message as the exception's detail * message, to indicate a test was canceled. * * @param message A message describing the cancellation. * @throws NullPointerException if message is null */ def cancel(message: String) = { if (message == null) throw new NullPointerException("message is null") throw newTestCanceledException(Some(message), None, 3) } /** * Throws TestCanceledException, with the passed * String message as the exception's detail * message and Throwable cause, to indicate a test failed. * * @param message A message describing the failure. * @param cause A Throwable that indicates the cause of the failure. * @throws NullPointerException if message or cause is null */ def cancel(message: String, cause: Throwable) = { if (message == null) throw new NullPointerException("message is null") if (cause == null) throw new NullPointerException("cause is null") throw newTestCanceledException(Some(message), Some(cause), 3) } /** * Throws TestCanceledException, with the passed * Throwable cause, to indicate a test failed. * The getMessage method of the thrown TestCanceledException * will return cause.toString. * * @param cause a Throwable that indicates the cause of the cancellation. * @throws NullPointerException if cause is null */ def cancel(cause: Throwable) = { if (cause == null) throw new NullPointerException("cause is null") throw newTestCanceledException(None, Some(cause), 3) } /** * Executes the block of code passed as the second parameter, and, if it * completes abruptly with a ModifiableMessage exception, * prepends the "clue" string passed as the first parameter to the beginning of the detail message * of that thrown exception, then rethrows it. If clue does not end in a white space * character, one space will be added * between it and the existing detail message (unless the detail message is * not defined). * *

* This method allows you to add more information about what went wrong that will be * reported when a test fails. Here's an example: *

* *
   * withClue("(Employee's name was: " + employee.name + ")") {
   *   intercept[IllegalArgumentException] {
   *     employee.getTask(-1)
   *   }
   * }
   * 
* *

* If an invocation of intercept completed abruptly with an exception, the resulting message would be something like: *

* *
   * (Employee's name was Bob Jones) Expected IllegalArgumentException to be thrown, but no exception was thrown
   * 
* * @throws NullPointerException if the passed clue is null */ def withClue[T](clue: Any)(fun: => T): T = { if (clue == null) throw new NullPointerException("clue was null") def prepend(currentMessage: Option[String]) = currentMessage match { case Some(msg) => if (clue.toString.last.isWhitespace) // TODO: shouldn't I also check if the head of msg isWhite? Some(clue.toString + msg) else Some(clue.toString + " " + msg) case None => Some(clue.toString) } try { fun } catch { case e: org.scalatest.exceptions.ModifiableMessage[_] => if (clue != "") throw e.modifyMessage(prepend) else throw e } } /* Hold off on this for now. See how people do with the simple one that takes an Any. def withClueFunction(sfun: Option[String] => Option[String])(fun: => Unit) { fun } */ } /** * Companion object that facilitates the importing of Assertions members as * an alternative to mixing it in. One use case is to import Assertions members so you can use * them in the Scala interpreter: * *
 * $scala -classpath scalatest.jar
 * Welcome to Scala version 2.7.3.final (Java HotSpot(TM) Client VM, Java 1.5.0_16).
 * Type in expressions to have them evaluated.
 * Type :help for more information.
 *  
 * scala> import org.scalatest.Assertions._
 * import org.scalatest.Assertions._
 *  
 * scala> assert(1 === 2)
 * org.scalatest.TestFailedException: 1 did not equal 2
 * 	at org.scalatest.Assertions$class.assert(Assertions.scala:211)
 * 	at org.scalatest.Assertions$.assert(Assertions.scala:511)
 * 	at .(:7)
 * 	at .()
 * 	at RequestResult$.(:3)
 * 	at RequestResult$.()
 * 	at RequestResult$result()
 * 	at sun.reflect.NativeMethodAccessorImpl.invoke...
 * 
 * scala> expect(3) { 1 + 3 }
 * org.scalatest.TestFailedException: Expected 3, but got 4
 * 	at org.scalatest.Assertions$class.expect(Assertions.scala:447)
 * 	at org.scalatest.Assertions$.expect(Assertions.scala:511)
 * 	at .(:7)
 * 	at .()
 * 	at RequestResult$.(:3)
 * 	at RequestResult$.()
 * 	at RequestResult$result()
 * 	at sun.reflect.NativeMethodAccessorImpl.in...
 * 
 * scala> val caught = intercept[StringIndexOutOfBoundsException] { "hi".charAt(-1) }
 * caught: StringIndexOutOfBoundsException = java.lang.StringIndexOutOfBoundsException: String index out of range: -1
 * 
* * @author Bill Venners */ object Assertions extends Assertions { private[scalatest] def areEqualComparingArraysStructurally(left: Any, right: Any) = { left match { case leftArray: Array[_] => right match { case rightArray: Array[_] => leftArray.deep.equals(rightArray.deep) case _ => left == right } case _ => left == right } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy