Futures (Scala)

Futures (Scala)

Introduction

In Akka, a Future is a data structure used to retrieve the result of some concurrent operation. This operation is usually performed by an Actor or by the Dispatcher directly. This result can be accessed synchronously (blocking) or asynchronously (non-blocking).

Execution Contexts

In order to execute callbacks and operations, Futures need something called an ExecutionContext, which is very similar to a java.util.concurrent.Executor. if you have an ActorSystem in scope, it will use its default dispatcher as the ExecutionContext, or you can use the factory methods provided by the ExecutionContext companion object to wrap Executors and ExecutorServices, or even create your own.

  1. import akka.dispatch.{ ExecutionContext, Promise }
  2.  
  3. implicit val ec = ExecutionContext.fromExecutorService(yourExecutorServiceGoesHere)
  4.  
  5. // Do stuff with your brand new shiny ExecutionContext
  6. val f = Promise.successful("foo")
  7.  
  8. // Then shut your ExecutionContext down at some
  9. // appropriate place in your program/application
  10. ec.shutdown()

Use With Actors

There are generally two ways of getting a reply from an Actor: the first is by a sent message (actor ! msg), which only works if the original sender was an Actor) and the second is through a Future.

Using an Actor's ? method to send a message will return a Future. To wait for and retrieve the actual result the simplest method is:

  1. import akka.dispatch.Await
  2. import akka.pattern.ask
  3. import akka.util.Timeout
  4. import akka.util.duration._
  5.  
  6. implicit val timeout = Timeout(5 seconds)
  7. val future = actor ? msg // enabled by the “ask” import
  8. val result = Await.result(future, timeout.duration).asInstanceOf[String]

This will cause the current thread to block and wait for the Actor to 'complete' the Future with it's reply. Blocking is discouraged though as it will cause performance problems. The blocking operations are located in Await.result and Await.ready to make it easy to spot where blocking occurs. Alternatives to blocking are discussed further within this documentation. Also note that the Future returned by an Actor is a Future[Any] since an Actor is dynamic. That is why the asInstanceOf is used in the above sample. When using non-blocking it is better to use the mapTo method to safely try to cast a Future to an expected type:

  1. import akka.dispatch.Future
  2. import akka.pattern.ask
  3.  
  4. val future: Future[String] = ask(actor, msg).mapTo[String]

The mapTo method will return a new Future that contains the result if the cast was successful, or a ClassCastException if not. Handling Exceptions will be discussed further within this documentation.

Use Directly

A common use case within Akka is to have some computation performed concurrently without needing the extra utility of an Actor. If you find yourself creating a pool of Actors for the sole reason of performing a calculation in parallel, there is an easier (and faster) way:

  1. import akka.dispatch.Await
  2. import akka.dispatch.Future
  3. import akka.util.duration._
  4.  
  5. val future = Future {
  6. "Hello" + "World"
  7. }
  8. val result = Await.result(future, 1 second)

In the above code the block passed to Future will be executed by the default Dispatcher, with the return value of the block used to complete the Future (in this case, the result would be the string: "HelloWorld"). Unlike a Future that is returned from an Actor, this Future is properly typed, and we also avoid the overhead of managing an Actor.

You can also create already completed Futures using the Promise companion, which can be either successes:

  1. val future = Promise.successful("Yay!")

Or failures:

  1. val otherFuture = Promise.failed[String](new IllegalArgumentException("Bang!"))

Functional Futures

Akka's Future has several monadic methods that are very similar to the ones used by Scala's collections. These allow you to create 'pipelines' or 'streams' that the result will travel through.

Future is a Monad

The first method for working with Future functionally is map. This method takes a Function which performs some operation on the result of the Future, and returning a new result. The return value of the map method is another Future that will contain the new result:

  1. val f1 = Future {
  2. "Hello" + "World"
  3. }
  4. val f2 = f1 map { x
  5. x.length
  6. }
  7. val result = Await.result(f2, 1 second)
  8. result must be(10)
  9. f1.value must be(Some(Right("HelloWorld")))

In this example we are joining two strings together within a Future. Instead of waiting for this to complete, we apply our function that calculates the length of the string using the map method. Now we have a second Future that will eventually contain an Int. When our original Future completes, it will also apply our function and complete the second Future with its result. When we finally get the result, it will contain the number 10. Our original Future still contains the string "HelloWorld" and is unaffected by the map.

The map method is fine if we are modifying a single Future, but if 2 or more Futures are involved map will not allow you to combine them together:

  1. val f1 = Future {
  2. "Hello" + "World"
  3. }
  4. val f2 = Promise.successful(3)
  5. val f3 = f1 map { x
  6. f2 map { y
  7. x.length * y
  8. }
  9. }

f3 is a Future[Future[Int]] instead of the desired Future[Int]. Instead, the flatMap method should be used:

  1. val f1 = Future {
  2. "Hello" + "World"
  3. }
  4. val f2 = Promise.successful(3)
  5. val f3 = f1 flatMap { x
  6. f2 map { y
  7. x.length * y
  8. }
  9. }
  10. val result = Await.result(f3, 1 second)
  11. result must be(30)

Composing futures using nested combinators it can sometimes become quite complicated and hard read, in these cases using Scala's 'for comprehensions' usually yields more readable code. See next section for examples.

If you need to do conditional propagation, you can use filter:

  1. val future1 = Promise.successful(4)
  2. val future2 = future1.filter(_ % 2 == 0)
  3. val result = Await.result(future2, 1 second)
  4. result must be(4)
  5.  
  6. val failedFilter = future1.filter(_ % 2 == 1).recover {
  7. case m: MatchError 0 //When filter fails, it will have a MatchError
  8. }
  9. val result2 = Await.result(failedFilter, 1 second)
  10. result2 must be(0) //Can only be 0 when there was a MatchError

For Comprehensions

Since Future has a map, filter and flatMap method it can be easily used in a 'for comprehension':

  1. val f = for {
  2. a Future(10 / 2) // 10 / 2 = 5
  3. b Future(a + 1) // 5 + 1 = 6
  4. c Future(a - 1) // 5 - 1 = 4
  5. if c > 3 // Future.filter
  6. } yield b * c // 6 * 4 = 24
  7.  
  8. // Note that the execution of futures a, b, and c
  9. // are not done in parallel.
  10.  
  11. val result = Await.result(f, 1 second)
  12. result must be(24)

Something to keep in mind when doing this is even though it looks like parts of the above example can run in parallel, each step of the for comprehension is run sequentially. This will happen on separate threads for each step but there isn't much benefit over running the calculations all within a single Future. The real benefit comes when the Futures are created first, and then combining them together.

Composing Futures

The example for comprehension above is an example of composing Futures. A common use case for this is combining the replies of several Actors into a single calculation without resorting to calling Await.result or Await.ready to block for each result. First an example of using Await.result:

  1. val f1 = ask(actor1, msg1)
  2. val f2 = ask(actor2, msg2)
  3.  
  4. val a = Await.result(f1, 1 second).asInstanceOf[Int]
  5. val b = Await.result(f2, 1 second).asInstanceOf[Int]
  6.  
  7. val f3 = ask(actor3, (a + b))
  8.  
  9. val result = Await.result(f3, 1 second).asInstanceOf[Int]

Here we wait for the results from the first 2 Actors before sending that result to the third Actor. We called Await.result 3 times, which caused our little program to block 3 times before getting our final result. Now compare that to this example:

  1. val f1 = ask(actor1, msg1)
  2. val f2 = ask(actor2, msg2)
  3.  
  4. val f3 = for {
  5. a f1.mapTo[Int]
  6. b f2.mapTo[Int]
  7. c ask(actor3, (a + b)).mapTo[Int]
  8. } yield c
  9.  
  10. val result = Await.result(f3, 1 second).asInstanceOf[Int]

Here we have 2 actors processing a single message each. Once the 2 results are available (note that we don't block to get these results!), they are being added together and sent to a third Actor, which replies with a string, which we assign to 'result'.

This is fine when dealing with a known amount of Actors, but can grow unwieldy if we have more then a handful. The sequence and traverse helper methods can make it easier to handle more complex use cases. Both of these methods are ways of turning, for a subclass T of Traversable, T[Future[A]] into a Future[T[A]]. For example:

  1. // oddActor returns odd numbers sequentially from 1 as a List[Future[Int]]
  2. val listOfFutures = List.fill(100)(akka.pattern.ask(oddActor, GetNext).mapTo[Int])
  3.  
  4. // now we have a Future[List[Int]]
  5. val futureList = Future.sequence(listOfFutures)
  6.  
  7. // Find the sum of the odd numbers
  8. val oddSum = Await.result(futureList.map(_.sum), 1 second).asInstanceOf[Int]
  9. oddSum must be(10000)

To better explain what happened in the example, Future.sequence is taking the List[Future[Int]] and turning it into a Future[List[Int]]. We can then use map to work with the List[Int] directly, and we find the sum of the List.

The traverse method is similar to sequence, but it takes a T[A] and a function A => Future[B] to return a Future[T[B]], where T is again a subclass of Traversable. For example, to use traverse to sum the first 100 odd numbers:

  1. val futureList = Future.traverse((1 to 100).toList)(x Future(x * 2 - 1))
  2. val oddSum = Await.result(futureList.map(_.sum), 1 second).asInstanceOf[Int]
  3. oddSum must be(10000)

This is the same result as this example:

  1. val futureList = Future.sequence((1 to 100).toList.map(x Future(x * 2 - 1)))
  2. val oddSum = Await.result(futureList.map(_.sum), 1 second).asInstanceOf[Int]
  3. oddSum must be(10000)

But it may be faster to use traverse as it doesn't have to create an intermediate List[Future[Int]].

Then there's a method that's called fold that takes a start-value, a sequence of Futures and a function from the type of the start-value and the type of the futures and returns something with the same type as the start-value, and then applies the function to all elements in the sequence of futures, asynchronously, the execution will start when the last of the Futures is completed.

  1. val futures = for (i 1 to 1000) yield Future(i * 2) // Create a sequence of Futures
  2. val futureSum = Future.fold(futures)(0)(_ + _)
  3. Await.result(futureSum, 1 second) must be(1001000)

That's all it takes!

If the sequence passed to fold is empty, it will return the start-value, in the case above, that will be 0. In some cases you don't have a start-value and you're able to use the value of the first completing Future in the sequence as the start-value, you can use reduce, it works like this:

  1. val futures = for (i 1 to 1000) yield Future(i * 2) // Create a sequence of Futures
  2. val futureSum = Future.reduce(futures)(_ + _)
  3. Await.result(futureSum, 1 second) must be(1001000)

Same as with fold, the execution will be done asynchronously when the last of the Future is completed, you can also parallelize it by chunking your futures into sub-sequences and reduce them, and then reduce the reduced results again.

Callbacks

Sometimes you just want to listen to a Future being completed, and react to that not by creating a new Future, but by side-effecting. For this Akka supports onComplete, onSuccess and onFailure, of which the latter two are specializations of the first.

  1. future onSuccess {
  2. case "bar" println("Got my bar alright!")
  3. case x: String println("Got some random string: " + x)
  4. }
  1. future onFailure {
  2. case ise: IllegalStateException if ise.getMessage == "OHNOES"
  3. //OHNOES! We are in deep trouble, do something!
  4. case e: Exception
  5. //Do something else
  6. }
  1. future onComplete {
  2. case Right(result) doSomethingOnSuccess(result)
  3. case Left(failure) doSomethingOnFailure(failure)
  4. }

Define Ordering

Since callbacks are executed in any order and potentially in parallel, it can be tricky at the times when you need sequential ordering of operations. But there's a solution and it's name is andThen. It creates a new Future with the specified callback, a Future that will have the same result as the Future it's called on, which allows for ordering like in the following sample:

  1. val result = Future { loadPage(url) } andThen {
  2. case Left(exception) log(exception)
  3. } andThen {
  4. case _ watchSomeTV
  5. }

Auxiliary Methods

Future fallbackTo combines 2 Futures into a new Future, and will hold the successful value of the second Future if the first Future fails.

  1. val future4 = future1 fallbackTo future2 fallbackTo future3

You can also combine two Futures into a new Future that will hold a tuple of the two Futures successful results, using the zip operation.

  1. val future3 = future1 zip future2 map { case (a, b) a + " " + b }

Exceptions

Since the result of a Future is created concurrently to the rest of the program, exceptions must be handled differently. It doesn't matter if an Actor or the dispatcher is completing the Future, if an Exception is caught the Future will contain it instead of a valid result. If a Future does contain an Exception, calling Await.result will cause it to be thrown again so it can be handled properly.

It is also possible to handle an Exception by returning a different result. This is done with the recover method. For example:

  1. val future = akka.pattern.ask(actor, msg1) recover {
  2. case e: ArithmeticException 0
  3. }

In this example, if the actor replied with a akka.actor.Status.Failure containing the ArithmeticException, our Future would have a result of 0. The recover method works very similarly to the standard try/catch blocks, so multiple Exceptions can be handled in this manner, and if an Exception is not handled this way it will behave as if we hadn't used the recover method.

You can also use the recoverWith method, which has the same relationship to recover as flatMap has to map, and is use like this:

  1. val future = akka.pattern.ask(actor, msg1) recoverWith {
  2. case e: ArithmeticException Promise.successful(0)
  3. case foo: IllegalArgumentException Promise.failed[Int](new IllegalStateException("All br0ken!"))
  4. }