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

org.scalatest.concurrent.JavaFutures.scala Maven / Gradle / Ivy

/*
 * Copyright 2001-2012 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.concurrent

import org.scalatest.time.Span
import java.util.concurrent.{TimeUnit, Future => FutureOfJava}
import org.scalatest.exceptions.StackDepthExceptionHelper.getStackDepthFun
import org.scalatest.Suite.anErrorThatShouldCauseAnAbort
import org.scalatest.Resources
import org.scalatest.exceptions.{TestPendingException, TestFailedException, TimeoutField}
import org.scalatest.exceptions.TestCanceledException

/**
 * Provides an implicit conversion from java.util.concurrent.Future[T] to
 * FutureConcept[T].
 *
 * 

* This trait enables you to invoke the methods defined on FutureConcept on a Java Future, as well as to pass a Java future * to the whenReady methods of supertrait Futures. * See the documentation for supertrait Futures for the details on the syntax this trait provides * for testing with Java futures. *

* * @author Bill Venners */ trait JavaFutures extends Futures { /** * Implicitly converts a java.util.concurrent.Future[T] to * FutureConcept[T], allowing you to invoke the methods * defined on FutureConcept on a Java Future, as well as to pass a Java future * to the whenReady methods of supertrait Futures. * *

* See the documentation for supertrait Futures for the details on the syntax this trait provides * for testing with Java futures. *

* *

If the get method of the underlying Java future throws java.util.concurrent.ExecutionException, and this * exception contains a non-null cause, that cause will be included in the TestFailedException as its cause. The ExecutionException * will be be included as the TestFailedException's cause only if the ExecutionException's cause is null. *

* *

* The isExpired method of the returned FutureConcept will always return false, because * the underlying type, java.util.concurrent.Future, does not support the notion of a timeout. The isCanceled * method of the returned FutureConcept will return the result of invoking isCancelled on the underlying * java.util.concurrent.Future. *

* * @param javaFuture a java.util.concurrent.Future[T] to convert * @return a FutureConcept[T] wrapping the passed java.util.concurrent.Future[T] */ implicit def convertJavaFuture[T](javaFuture: FutureOfJava[T]): FutureConcept[T] = new FutureConcept[T] { def eitherValue: Option[Either[Throwable, T]] = if (javaFuture.isDone()) Some(Right(javaFuture.get)) else None def isExpired: Boolean = false // Java Futures don't support the notion of a timeout def isCanceled: Boolean = javaFuture.isCancelled // Two ll's in Canceled. The verbosity of Java strikes again! // TODO: Catch TimeoutException and wrap that in a TFE with ScalaTest's TimeoutException I think. // def awaitAtMost(span: Span): T = javaFuture.get(span.totalNanos, TimeUnit.NANOSECONDS) override def futureValue(implicit config: PatienceConfig): T = { val st = Thread.currentThread.getStackTrace val callerStackFrame = if (!st(2).getMethodName.contains("futureValue")) st(2) else st(3) val methodName = if (callerStackFrame.getFileName == "Futures.scala" && callerStackFrame.getMethodName == "whenReady") "whenReady" else if (callerStackFrame.getFileName == "Futures.scala" && callerStackFrame.getMethodName == "isReadyWithin") "isReadyWithin" else "futureValue" val adjustment = if (methodName == "whenReady") 3 else 0 if (javaFuture.isCanceled) throw new TestFailedException( sde => Some(Resources("futureWasCanceled")), None, getStackDepthFun("JavaFutures.scala", methodName, adjustment) ) try { javaFuture.get(config.timeout.totalNanos, TimeUnit.NANOSECONDS) } catch { case e: java.util.concurrent.TimeoutException => throw new TestFailedException( sde => Some(Resources("wasNeverReady")), None, getStackDepthFun("JavaFutures.scala", methodName, adjustment) ) with TimeoutField { val timeout: Span = config.timeout } case e: java.util.concurrent.ExecutionException => val cause = e.getCause val exToReport = if (cause == null) e else cause if (anErrorThatShouldCauseAnAbort(exToReport) || exToReport.isInstanceOf[TestPendingException] || exToReport.isInstanceOf[TestCanceledException]) { throw exToReport } throw new TestFailedException( sde => Some { if (exToReport.getMessage == null) Resources("futureReturnedAnException", exToReport.getClass.getName) else Resources("futureReturnedAnExceptionWithMessage", exToReport.getClass.getName, exToReport.getMessage) }, Some(exToReport), getStackDepthFun("JavaFutures.scala", methodName, adjustment) ) } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy