2012-01-17 17:04:20 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Copyright (C) 2009-2011 Typesafe Inc. <http://www.typesafe.com>
|
|
|
|
|
|
*/
|
2012-01-18 10:18:51 +01:00
|
|
|
|
package akka.pattern
|
2012-01-17 17:04:20 +01:00
|
|
|
|
|
|
|
|
|
|
object Patterns {
|
|
|
|
|
|
import akka.actor.ActorRef
|
|
|
|
|
|
import akka.dispatch.Future
|
2012-01-18 10:18:51 +01:00
|
|
|
|
import akka.pattern.{ ask ⇒ scalaAsk }
|
2012-01-17 17:04:20 +01:00
|
|
|
|
import akka.util.Timeout
|
|
|
|
|
|
|
2012-01-18 11:52:35 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[akka.dispatch.Future]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
|
|
|
|
|
* will be completed with an [[akka.actor.AskTimeoutException]] after the
|
|
|
|
|
|
* given timeout has expired; this is independent from any timeout applied
|
|
|
|
|
|
* while awaiting a result for this future (i.e. in
|
|
|
|
|
|
* `Await.result(..., timeout)`).
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>This variant will use the `akka.actor.timeout` from the configuration.</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Warning:</b>
|
|
|
|
|
|
* When using future callbacks, inside actors you need to carefully avoid closing over
|
|
|
|
|
|
* the containing actor’s object, i.e. do not call methods or access mutable state
|
|
|
|
|
|
* on the enclosing actor from within the callback. This would break the actor
|
|
|
|
|
|
* encapsulation and may introduce synchronization bugs and race conditions because
|
|
|
|
|
|
* the callback will be scheduled concurrently to the enclosing actor. Unfortunately
|
|
|
|
|
|
* there is not yet a way to detect these illegal accesses at compile time.
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Recommended usage:</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
|
|
|
|
|
* final Future<Object> f = Patterns.ask(worker, request, timeout);
|
|
|
|
|
|
* f.onSuccess(new Procedure<Object>() {
|
|
|
|
|
|
* public void apply(Object o) {
|
|
|
|
|
|
* nextActor.tell(new EnrichedResult(request, o));
|
|
|
|
|
|
* }
|
|
|
|
|
|
* });
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(actor: ActorRef, message: Any): Future[AnyRef] = scalaAsk(actor, message).asInstanceOf[Future[AnyRef]]
|
|
|
|
|
|
|
2012-01-17 17:04:20 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[akka.dispatch.Future]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
|
|
|
|
|
* will be completed with an [[akka.actor.AskTimeoutException]] after the
|
|
|
|
|
|
* given timeout has expired; this is independent from any timeout applied
|
|
|
|
|
|
* while awaiting a result for this future (i.e. in
|
|
|
|
|
|
* `Await.result(..., timeout)`).
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Warning:</b>
|
|
|
|
|
|
* When using future callbacks, inside actors you need to carefully avoid closing over
|
|
|
|
|
|
* the containing actor’s object, i.e. do not call methods or access mutable state
|
|
|
|
|
|
* on the enclosing actor from within the callback. This would break the actor
|
|
|
|
|
|
* encapsulation and may introduce synchronization bugs and race conditions because
|
|
|
|
|
|
* the callback will be scheduled concurrently to the enclosing actor. Unfortunately
|
|
|
|
|
|
* there is not yet a way to detect these illegal accesses at compile time.
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Recommended usage:</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
|
|
|
|
|
* final Future<Object> f = Patterns.ask(worker, request, timeout);
|
|
|
|
|
|
* f.onSuccess(new Procedure<Object>() {
|
|
|
|
|
|
* public void apply(Object o) {
|
|
|
|
|
|
* nextActor.tell(new EnrichedResult(request, o));
|
|
|
|
|
|
* }
|
|
|
|
|
|
* });
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(actor: ActorRef, message: Any, timeout: Timeout): Future[AnyRef] = scalaAsk(actor, message)(timeout).asInstanceOf[Future[AnyRef]]
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[akka.dispatch.Future]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
|
|
|
|
|
* will be completed with an [[akka.actor.AskTimeoutException]] after the
|
|
|
|
|
|
* given timeout has expired; this is independent from any timeout applied
|
|
|
|
|
|
* while awaiting a result for this future (i.e. in
|
|
|
|
|
|
* `Await.result(..., timeout)`).
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Warning:</b>
|
|
|
|
|
|
* When using future callbacks, inside actors you need to carefully avoid closing over
|
|
|
|
|
|
* the containing actor’s object, i.e. do not call methods or access mutable state
|
|
|
|
|
|
* on the enclosing actor from within the callback. This would break the actor
|
|
|
|
|
|
* encapsulation and may introduce synchronization bugs and race conditions because
|
|
|
|
|
|
* the callback will be scheduled concurrently to the enclosing actor. Unfortunately
|
|
|
|
|
|
* there is not yet a way to detect these illegal accesses at compile time.
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Recommended usage:</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
|
|
|
|
|
* final Future<Object> f = Patterns.ask(worker, request, timeout);
|
|
|
|
|
|
* f.onSuccess(new Procedure<Object>() {
|
|
|
|
|
|
* public void apply(Object o) {
|
|
|
|
|
|
* nextActor.tell(new EnrichedResult(request, o));
|
|
|
|
|
|
* }
|
|
|
|
|
|
* });
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(actor: ActorRef, message: Any, timeoutMillis: Long): Future[AnyRef] = scalaAsk(actor, message)(new Timeout(timeoutMillis)).asInstanceOf[Future[AnyRef]]
|
2012-01-18 13:01:24 +01:00
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Register an onComplete callback on this [[akka.dispatch.Future]] to send
|
|
|
|
|
|
* the result to the given actor reference. Returns the original Future to
|
|
|
|
|
|
* allow method chaining.
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Recommended usage example:</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
|
|
|
|
|
* val f = ask(worker, request)(timeout)
|
|
|
|
|
|
* flow {
|
|
|
|
|
|
* EnrichedRequest(request, f())
|
|
|
|
|
|
* } pipeTo nextActor
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*
|
|
|
|
|
|
* [see [[akka.dispatch.Future]] for a description of `flow`]
|
|
|
|
|
|
*/
|
|
|
|
|
|
def pipeTo[T](future: Future[T], actorRef: ActorRef): Future[T] = akka.pattern.pipeTo(future, actorRef)
|
2012-01-18 10:18:51 +01:00
|
|
|
|
}
|