2012-01-17 17:04:20 +01:00
|
|
|
|
/**
|
2018-01-04 17:26:29 +00:00
|
|
|
|
* Copyright (C) 2009-2018 Lightbend Inc. <https://www.lightbend.com>
|
2012-01-17 17:04:20 +01:00
|
|
|
|
*/
|
2018-03-13 23:45:55 +09:00
|
|
|
|
|
2012-01-18 10:18:51 +01:00
|
|
|
|
package akka.pattern
|
2012-01-17 17:04:20 +01:00
|
|
|
|
|
2017-08-02 14:33:47 +09:00
|
|
|
|
import java.util.concurrent.{ Callable, CompletionStage, TimeUnit }
|
|
|
|
|
|
|
2013-07-15 15:43:22 +02:00
|
|
|
|
import akka.actor.{ ActorSelection, Scheduler }
|
2018-04-11 16:47:36 +02:00
|
|
|
|
import akka.util.JavaDurationConverters._
|
2017-08-02 14:33:47 +09:00
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
import scala.compat.java8.FutureConverters._
|
2017-08-02 14:33:47 +09:00
|
|
|
|
import scala.concurrent.ExecutionContext
|
2012-04-25 18:34:16 +02:00
|
|
|
|
|
2017-08-02 14:33:47 +09:00
|
|
|
|
/**
|
|
|
|
|
|
* "Pre Java 8" Java API for Akka patterns such as `ask`, `pipe` and others.
|
|
|
|
|
|
*
|
|
|
|
|
|
* These methods are possible to call from Java however work with the Scala [[scala.concurrent.Future]],
|
|
|
|
|
|
* due to the lack of non-blocking reactive Future implementation before Java 8.
|
|
|
|
|
|
*
|
|
|
|
|
|
* For Java applications developed with Java 8 and later, you might want to use [[akka.pattern.PatternsCS]] instead,
|
|
|
|
|
|
* which provide alternatives for these patterns which work with [[java.util.concurrent.CompletionStage]].
|
|
|
|
|
|
*/
|
2012-01-17 17:04:20 +01:00
|
|
|
|
object Patterns {
|
2017-08-02 14:33:47 +09:00
|
|
|
|
import akka.actor.ActorRef
|
2015-10-25 14:38:10 +03:00
|
|
|
|
import akka.japi
|
2018-03-26 14:56:20 +03:00
|
|
|
|
import akka.pattern.{
|
|
|
|
|
|
after ⇒ scalaAfter,
|
|
|
|
|
|
ask ⇒ scalaAsk,
|
|
|
|
|
|
gracefulStop ⇒ scalaGracefulStop,
|
|
|
|
|
|
pipe ⇒ scalaPipe,
|
|
|
|
|
|
retry ⇒ scalaRetry
|
|
|
|
|
|
}
|
2012-06-29 13:33:20 +02:00
|
|
|
|
import akka.util.Timeout
|
2017-08-02 14:33:47 +09:00
|
|
|
|
|
2012-07-04 15:25:30 +02:00
|
|
|
|
import scala.concurrent.Future
|
2015-10-25 14:38:10 +03:00
|
|
|
|
import scala.concurrent.duration._
|
2012-01-17 17:04:20 +01:00
|
|
|
|
|
2012-01-18 11:52:35 +01:00
|
|
|
|
/**
|
2012-01-18 13:26:11 +01:00
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
2012-07-04 15:25:30 +02:00
|
|
|
|
* Sends a message asynchronously and returns a [[scala.concurrent.Future]]
|
2012-01-18 11:52:35 +01:00
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
2012-05-18 15:04:08 +02:00
|
|
|
|
* will be completed with an [[akka.pattern.AskTimeoutException]] after the
|
2012-01-17 17:04:20 +01:00
|
|
|
|
* 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]]
|
|
|
|
|
|
|
2015-10-25 14:38:10 +03:00
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final Future<Object> f = Patterns.askWithReplyTo(
|
2015-10-25 14:38:10 +03:00
|
|
|
|
* worker,
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* replyTo -> new Request(replyTo),
|
2015-10-25 14:38:10 +03:00
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2018-04-11 13:49:42 +02:00
|
|
|
|
def askWithReplyTo(actor: ActorRef, messageFactory: japi.Function[ActorRef, Any], timeout: Timeout): Future[AnyRef] =
|
|
|
|
|
|
extended.ask(actor, messageFactory.apply _)(timeout).asInstanceOf[Future[AnyRef]]
|
2015-10-25 14:38:10 +03:00
|
|
|
|
|
2012-01-17 17:04:20 +01:00
|
|
|
|
/**
|
2012-01-18 13:26:11 +01:00
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
2012-07-04 15:25:30 +02:00
|
|
|
|
* Sends a message asynchronously and returns a [[scala.concurrent.Future]]
|
2012-01-17 17:04:20 +01:00
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
2012-05-18 15:04:08 +02:00
|
|
|
|
* will be completed with an [[akka.pattern.AskTimeoutException]] after the
|
2012-01-17 17:04:20 +01:00
|
|
|
|
* 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));
|
|
|
|
|
|
* }
|
|
|
|
|
|
* });
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2013-07-15 15:43:22 +02:00
|
|
|
|
def ask(actor: ActorRef, message: Any, timeoutMillis: Long): Future[AnyRef] =
|
2014-03-12 15:00:27 +01:00
|
|
|
|
scalaAsk(actor, message)(new Timeout(timeoutMillis, TimeUnit.MILLISECONDS)).asInstanceOf[Future[AnyRef]]
|
2013-07-15 15:43:22 +02:00
|
|
|
|
|
2015-10-25 14:38:10 +03:00
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final Future<Object> f = Patterns.askWithReplyTo(
|
2015-10-25 14:38:10 +03:00
|
|
|
|
* worker,
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* replyTo -> new Request(replyTo),
|
2015-10-25 14:38:10 +03:00
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2018-04-11 13:49:42 +02:00
|
|
|
|
def askWithReplyTo(actor: ActorRef, messageFactory: japi.Function[ActorRef, Any], timeoutMillis: Long): Future[AnyRef] =
|
|
|
|
|
|
extended.ask(actor, messageFactory.apply _)(Timeout(timeoutMillis.millis)).asInstanceOf[Future[AnyRef]]
|
2015-10-25 14:38:10 +03:00
|
|
|
|
|
2013-07-15 15:43:22 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[scala.concurrent.Future]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target [[akka.actor.ActorSelection]]
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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(selection, request, timeout);
|
|
|
|
|
|
* f.onSuccess(new Procedure<Object>() {
|
|
|
|
|
|
* public void apply(Object o) {
|
|
|
|
|
|
* nextActor.tell(new EnrichedResult(request, o));
|
|
|
|
|
|
* }
|
|
|
|
|
|
* });
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(selection: ActorSelection, message: Any, timeout: Timeout): Future[AnyRef] =
|
|
|
|
|
|
scalaAsk(selection, message)(timeout).asInstanceOf[Future[AnyRef]]
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[scala.concurrent.Future]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target [[akka.actor.ActorSelection]]
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The Future
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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(selection, request, timeout);
|
|
|
|
|
|
* f.onSuccess(new Procedure<Object>() {
|
|
|
|
|
|
* public void apply(Object o) {
|
|
|
|
|
|
* nextActor.tell(new EnrichedResult(request, o));
|
|
|
|
|
|
* }
|
|
|
|
|
|
* });
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(selection: ActorSelection, message: Any, timeoutMillis: Long): Future[AnyRef] =
|
2014-03-12 15:00:27 +01:00
|
|
|
|
scalaAsk(selection, message)(new Timeout(timeoutMillis, TimeUnit.MILLISECONDS)).asInstanceOf[Future[AnyRef]]
|
2015-10-25 14:38:10 +03:00
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final Future<Object> f = Patterns.askWithReplyTo(
|
2015-10-25 14:38:10 +03:00
|
|
|
|
* selection,
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* replyTo -> new Request(replyTo),
|
2015-10-25 14:38:10 +03:00
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2018-04-11 13:49:42 +02:00
|
|
|
|
def askWithReplyTo(selection: ActorSelection, messageFactory: japi.Function[ActorRef, Any], timeoutMillis: Long): Future[AnyRef] =
|
|
|
|
|
|
extended.ask(selection, messageFactory.apply _)(Timeout(timeoutMillis.millis)).asInstanceOf[Future[AnyRef]]
|
2012-01-18 13:01:24 +01:00
|
|
|
|
|
|
|
|
|
|
/**
|
2012-07-04 15:25:30 +02:00
|
|
|
|
* Register an onComplete callback on this [[scala.concurrent.Future]] to send
|
2013-06-25 16:17:13 +02:00
|
|
|
|
* the result to the given [[akka.actor.ActorRef]] or [[akka.actor.ActorSelection]].
|
|
|
|
|
|
* Returns the original Future to allow method chaining.
|
2013-08-23 09:07:38 +02:00
|
|
|
|
* If the future was completed with failure it is sent as a [[akka.actor.Status.Failure]]
|
|
|
|
|
|
* to the recipient.
|
2012-01-18 13:01:24 +01:00
|
|
|
|
*
|
|
|
|
|
|
* <b>Recommended usage example:</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2012-01-23 11:58:27 +01:00
|
|
|
|
* final Future<Object> f = Patterns.ask(worker, request, timeout);
|
|
|
|
|
|
* // apply some transformation (i.e. enrich with request info)
|
|
|
|
|
|
* final Future<Object> transformed = f.map(new akka.japi.Function<Object, Object>() { ... });
|
2018-06-09 17:42:56 +09:00
|
|
|
|
* // send it on to the next operator
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* Patterns.pipe(transformed, context).to(nextActor);
|
2012-01-18 13:01:24 +01:00
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2012-07-17 17:21:08 +02:00
|
|
|
|
def pipe[T](future: Future[T], context: ExecutionContext): PipeableFuture[T] = scalaPipe(future)(context)
|
2012-01-01 20:48:03 +01:00
|
|
|
|
|
|
|
|
|
|
/**
|
2012-07-04 15:25:30 +02:00
|
|
|
|
* Returns a [[scala.concurrent.Future]] that will be completed with success (value `true`) when
|
2012-01-01 20:48:03 +01:00
|
|
|
|
* existing messages of the target actor has been processed and the actor has been
|
|
|
|
|
|
* terminated.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Useful when you need to wait for termination or compose ordered termination of several actors.
|
|
|
|
|
|
*
|
2012-07-04 15:25:30 +02:00
|
|
|
|
* If the target actor isn't terminated within the timeout the [[scala.concurrent.Future]]
|
2012-05-18 15:04:08 +02:00
|
|
|
|
* is completed with failure [[akka.pattern.AskTimeoutException]].
|
2012-01-01 20:48:03 +01:00
|
|
|
|
*/
|
2013-03-30 01:03:17 +01:00
|
|
|
|
def gracefulStop(target: ActorRef, timeout: FiniteDuration): Future[java.lang.Boolean] =
|
|
|
|
|
|
scalaGracefulStop(target, timeout).asInstanceOf[Future[java.lang.Boolean]]
|
2012-04-25 18:34:16 +02:00
|
|
|
|
|
2014-01-21 17:15:09 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[scala.concurrent.Future]] that will be completed with success (value `true`) when
|
|
|
|
|
|
* existing messages of the target actor has been processed and the actor has been
|
|
|
|
|
|
* terminated.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Useful when you need to wait for termination or compose ordered termination of several actors.
|
|
|
|
|
|
*
|
|
|
|
|
|
* If you want to invoke specialized stopping logic on your target actor instead of PoisonPill, you can pass your
|
|
|
|
|
|
* stop command as `stopMessage` parameter
|
|
|
|
|
|
*
|
|
|
|
|
|
* If the target actor isn't terminated within the timeout the [[scala.concurrent.Future]]
|
|
|
|
|
|
* is completed with failure [[akka.pattern.AskTimeoutException]].
|
|
|
|
|
|
*/
|
|
|
|
|
|
def gracefulStop(target: ActorRef, timeout: FiniteDuration, stopMessage: Any): Future[java.lang.Boolean] =
|
|
|
|
|
|
scalaGracefulStop(target, timeout, stopMessage).asInstanceOf[Future[java.lang.Boolean]]
|
|
|
|
|
|
|
2012-04-25 18:34:16 +02:00
|
|
|
|
/**
|
2012-07-04 15:25:30 +02:00
|
|
|
|
* Returns a [[scala.concurrent.Future]] that will be completed with the success or failure of the provided Callable
|
2012-04-25 18:34:16 +02:00
|
|
|
|
* after the specified duration.
|
|
|
|
|
|
*/
|
2012-09-18 09:58:30 +02:00
|
|
|
|
def after[T](duration: FiniteDuration, scheduler: Scheduler, context: ExecutionContext, value: Callable[Future[T]]): Future[T] =
|
2012-04-25 18:34:16 +02:00
|
|
|
|
scalaAfter(duration, scheduler)(value.call())(context)
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* Returns a [[scala.concurrent.Future]] that will be completed with the success or failure of the provided Callable
|
2012-04-25 18:34:16 +02:00
|
|
|
|
* after the specified duration.
|
|
|
|
|
|
*/
|
2012-09-18 09:58:30 +02:00
|
|
|
|
def after[T](duration: FiniteDuration, scheduler: Scheduler, context: ExecutionContext, value: Future[T]): Future[T] =
|
2012-04-25 18:34:16 +02:00
|
|
|
|
scalaAfter(duration, scheduler)(value)(context)
|
2018-03-26 14:56:20 +03:00
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Returns an internally retrying [[scala.concurrent.Future]]
|
|
|
|
|
|
* The first attempt will be made immediately, and each subsequent attempt will be made after 'delay'.
|
|
|
|
|
|
* A scheduler (eg context.system.scheduler) must be provided to delay each retry
|
|
|
|
|
|
* If attempts are exhausted the returned future is simply the result of invoking attempt.
|
|
|
|
|
|
* Note that the attempt function will be invoked on the given execution context for subsequent tries and
|
|
|
|
|
|
* therefore must be thread safe (not touch unsafe mutable state).
|
|
|
|
|
|
*/
|
|
|
|
|
|
def retry[T](attempt: Callable[Future[T]], attempts: Int, delay: FiniteDuration, scheduler: Scheduler,
|
|
|
|
|
|
context: ExecutionContext): Future[T] =
|
|
|
|
|
|
scalaRetry(() ⇒ attempt.call, attempts, delay)(context, scheduler)
|
2012-01-18 10:18:51 +01:00
|
|
|
|
}
|
2016-01-21 16:37:26 +01:00
|
|
|
|
|
2017-08-02 14:33:47 +09:00
|
|
|
|
/**
|
|
|
|
|
|
* Java 8+ API for Akka patterns such as `ask`, `pipe` and others which work with [[java.util.concurrent.CompletionStage]].
|
|
|
|
|
|
*
|
|
|
|
|
|
* For working with Scala [[scala.concurrent.Future]] from Java you may want to use [[akka.pattern.Patterns]] instead.
|
|
|
|
|
|
*/
|
2016-01-21 16:37:26 +01:00
|
|
|
|
object PatternsCS {
|
2017-08-02 14:33:47 +09:00
|
|
|
|
import akka.actor.ActorRef
|
2016-01-21 16:37:26 +01:00
|
|
|
|
import akka.japi
|
2018-03-26 14:56:20 +03:00
|
|
|
|
import akka.pattern.{ ask ⇒ scalaAsk, gracefulStop ⇒ scalaGracefulStop, retry ⇒ scalaRetry }
|
2016-01-21 16:37:26 +01:00
|
|
|
|
import akka.util.Timeout
|
2017-08-02 14:33:47 +09:00
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
import scala.concurrent.duration._
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The CompletionStage
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.ask(worker, request, timeout);
|
|
|
|
|
|
* f.thenRun(result -> nextActor.tell(new EnrichedResult(request, result)));
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2018-06-27 01:12:11 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.15")
|
2016-01-21 16:37:26 +01:00
|
|
|
|
def ask(actor: ActorRef, message: Any, timeout: Timeout): CompletionStage[AnyRef] =
|
|
|
|
|
|
scalaAsk(actor, message)(timeout).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
|
|
|
|
|
|
2018-06-27 01:12:11 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The CompletionStage
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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 CompletionStage<Object> f = PatternsCS.ask(worker, request, duration);
|
|
|
|
|
|
* f.thenRun(result -> nextActor.tell(new EnrichedResult(request, result)));
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(actor: ActorRef, message: Any, timeout: java.time.Duration): CompletionStage[AnyRef] =
|
|
|
|
|
|
ask(actor, message, Timeout.create(timeout))
|
|
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.askWithReplyTo(
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* worker,
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* askSender -> new Request(askSender),
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
2018-02-22 19:51:34 +00:00
|
|
|
|
*
|
|
|
|
|
|
* @param actor the actor to be asked
|
|
|
|
|
|
* @param messageFactory function taking an actor ref and returning the message to be sent
|
2018-06-09 17:42:56 +09:00
|
|
|
|
* @param timeout the timeout for the response before failing the returned completion operator
|
2016-01-21 16:37:26 +01:00
|
|
|
|
*/
|
2018-06-27 01:12:11 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.15")
|
2018-04-11 13:49:42 +02:00
|
|
|
|
def askWithReplyTo(actor: ActorRef, messageFactory: japi.function.Function[ActorRef, Any], timeout: Timeout): CompletionStage[AnyRef] =
|
|
|
|
|
|
extended.ask(actor, messageFactory.apply _)(timeout).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
2016-01-21 16:37:26 +01:00
|
|
|
|
|
2018-06-27 01:12:11 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
|
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.askWithReplyTo(
|
|
|
|
|
|
* worker,
|
|
|
|
|
|
* askSender -> new Request(askSender),
|
|
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*
|
|
|
|
|
|
* @param actor the actor to be asked
|
|
|
|
|
|
* @param messageFactory function taking an actor ref and returning the message to be sent
|
|
|
|
|
|
* @param timeout the timeout for the response before failing the returned completion stage
|
|
|
|
|
|
*/
|
|
|
|
|
|
def askWithReplyTo(actor: ActorRef, messageFactory: japi.function.Function[ActorRef, Any], timeout: java.time.Duration): CompletionStage[AnyRef] =
|
|
|
|
|
|
extended.ask(actor, messageFactory.apply _)(Timeout.create(timeout)).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
|
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target actor
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The CompletionStage
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.ask(worker, request, timeout);
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* f.thenRun(result -> nextActor.tell(new EnrichedResult(request, result)));
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(actor: ActorRef, message: Any, timeoutMillis: Long): CompletionStage[AnyRef] =
|
|
|
|
|
|
scalaAsk(actor, message)(new Timeout(timeoutMillis, TimeUnit.MILLISECONDS)).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.askWithReplyTo(
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* worker,
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* replyTo -> new Request(replyTo),
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
2018-02-22 19:51:34 +00:00
|
|
|
|
*
|
|
|
|
|
|
* @param actor the actor to be asked
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* @param messageFactory function taking an actor ref to reply to and returning the message to be sent
|
2018-06-09 17:42:56 +09:00
|
|
|
|
* @param timeoutMillis the timeout for the response before failing the returned completion operator
|
2016-01-21 16:37:26 +01:00
|
|
|
|
*/
|
2018-04-11 13:49:42 +02:00
|
|
|
|
def askWithReplyTo(actor: ActorRef, messageFactory: japi.function.Function[ActorRef, Any], timeoutMillis: Long): CompletionStage[AnyRef] =
|
|
|
|
|
|
askWithReplyTo(actor, messageFactory, Timeout(timeoutMillis.millis))
|
2016-01-21 16:37:26 +01:00
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target [[akka.actor.ActorSelection]]
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The CompletionStage
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.ask(selection, request, timeout);
|
|
|
|
|
|
* f.thenRun(result -> nextActor.tell(new EnrichedResult(request, result)));
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2018-06-27 01:12:11 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.15")
|
2016-01-21 16:37:26 +01:00
|
|
|
|
def ask(selection: ActorSelection, message: Any, timeout: Timeout): CompletionStage[AnyRef] =
|
|
|
|
|
|
scalaAsk(selection, message)(timeout).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target [[akka.actor.ActorSelection]]
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The CompletionStage
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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
|
2018-06-27 01:12:11 +02:00
|
|
|
|
* 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 CompletionStage<Object> f = PatternsCS.ask(selection, request, duration);
|
|
|
|
|
|
* f.thenRun(result -> nextActor.tell(new EnrichedResult(request, result)));
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(selection: ActorSelection, message: Any, timeout: java.time.Duration): CompletionStage[AnyRef] =
|
|
|
|
|
|
ask(selection, message, Timeout.create(timeout))
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* <i>Java API for `akka.pattern.ask`:</i>
|
|
|
|
|
|
* Sends a message asynchronously and returns a [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* holding the eventual reply message; this means that the target [[akka.actor.ActorSelection]]
|
|
|
|
|
|
* needs to send the result to the `sender` reference provided. The CompletionStage
|
|
|
|
|
|
* will be completed with an [[akka.pattern.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
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* 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>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.ask(selection, request, timeout);
|
|
|
|
|
|
* f.thenRun(result -> nextActor.tell(new EnrichedResult(request, result)));
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def ask(selection: ActorSelection, message: Any, timeoutMillis: Long): CompletionStage[AnyRef] =
|
|
|
|
|
|
scalaAsk(selection, message)(new Timeout(timeoutMillis, TimeUnit.MILLISECONDS)).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* A variation of ask which allows to implement "replyTo" pattern by including
|
|
|
|
|
|
* sender reference in message.
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* final CompletionStage<Object> f = Patterns.askWithReplyTo(
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* selection,
|
2018-04-11 13:49:42 +02:00
|
|
|
|
* askSender -> new Request(askSender),
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* timeout);
|
|
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
2018-04-11 13:49:42 +02:00
|
|
|
|
def askWithReplyTo(selection: ActorSelection, messageFactory: japi.Function[ActorRef, Any], timeoutMillis: Long): CompletionStage[AnyRef] =
|
|
|
|
|
|
extended.ask(selection, messageFactory.apply _)(Timeout(timeoutMillis.millis)).toJava.asInstanceOf[CompletionStage[AnyRef]]
|
2016-01-21 16:37:26 +01:00
|
|
|
|
|
|
|
|
|
|
/**
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* When this [[java.util.concurrent.CompletionStage]] finishes, send its result to the given
|
|
|
|
|
|
* [[akka.actor.ActorRef]] or [[akka.actor.ActorSelection]].
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* Returns the original CompletionStage to allow method chaining.
|
|
|
|
|
|
* If the future was completed with failure it is sent as a [[akka.actor.Status.Failure]]
|
|
|
|
|
|
* to the recipient.
|
|
|
|
|
|
*
|
|
|
|
|
|
* <b>Recommended usage example:</b>
|
|
|
|
|
|
*
|
|
|
|
|
|
* {{{
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* final CompletionStage<Object> f = PatternsCS.ask(worker, request, timeout);
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* // apply some transformation (i.e. enrich with request info)
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* final CompletionStage<Object> transformed = f.thenApply(result -> { ... });
|
2018-06-09 17:42:56 +09:00
|
|
|
|
* // send it on to the next operator
|
2018-05-10 20:32:28 +08:00
|
|
|
|
* PatternsCS.pipe(transformed, context).to(nextActor);
|
2016-01-21 16:37:26 +01:00
|
|
|
|
* }}}
|
|
|
|
|
|
*/
|
|
|
|
|
|
def pipe[T](future: CompletionStage[T], context: ExecutionContext): PipeableCompletionStage[T] = pipeCompletionStage(future)(context)
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with success (value `true`) when
|
|
|
|
|
|
* existing messages of the target actor has been processed and the actor has been
|
|
|
|
|
|
* terminated.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Useful when you need to wait for termination or compose ordered termination of several actors.
|
|
|
|
|
|
*
|
|
|
|
|
|
* If the target actor isn't terminated within the timeout the [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* is completed with failure [[akka.pattern.AskTimeoutException]].
|
|
|
|
|
|
*/
|
2018-04-11 16:47:36 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.12")
|
2016-01-21 16:37:26 +01:00
|
|
|
|
def gracefulStop(target: ActorRef, timeout: FiniteDuration): CompletionStage[java.lang.Boolean] =
|
|
|
|
|
|
scalaGracefulStop(target, timeout).toJava.asInstanceOf[CompletionStage[java.lang.Boolean]]
|
|
|
|
|
|
|
2018-04-11 16:47:36 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with success (value `true`) when
|
|
|
|
|
|
* existing messages of the target actor has been processed and the actor has been
|
|
|
|
|
|
* terminated.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Useful when you need to wait for termination or compose ordered termination of several actors.
|
|
|
|
|
|
*
|
|
|
|
|
|
* If the target actor isn't terminated within the timeout the [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* is completed with failure [[akka.pattern.AskTimeoutException]].
|
|
|
|
|
|
*/
|
|
|
|
|
|
def gracefulStop(target: ActorRef, timeout: java.time.Duration): CompletionStage[java.lang.Boolean] =
|
|
|
|
|
|
scalaGracefulStop(target, timeout.asScala).toJava.asInstanceOf[CompletionStage[java.lang.Boolean]]
|
|
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with success (value `true`) when
|
|
|
|
|
|
* existing messages of the target actor has been processed and the actor has been
|
|
|
|
|
|
* terminated.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Useful when you need to wait for termination or compose ordered termination of several actors.
|
|
|
|
|
|
*
|
|
|
|
|
|
* If you want to invoke specialized stopping logic on your target actor instead of PoisonPill, you can pass your
|
|
|
|
|
|
* stop command as `stopMessage` parameter
|
|
|
|
|
|
*
|
|
|
|
|
|
* If the target actor isn't terminated within the timeout the [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* is completed with failure [[akka.pattern.AskTimeoutException]].
|
|
|
|
|
|
*/
|
2018-04-11 16:47:36 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.12")
|
2016-01-21 16:37:26 +01:00
|
|
|
|
def gracefulStop(target: ActorRef, timeout: FiniteDuration, stopMessage: Any): CompletionStage[java.lang.Boolean] =
|
|
|
|
|
|
scalaGracefulStop(target, timeout, stopMessage).toJava.asInstanceOf[CompletionStage[java.lang.Boolean]]
|
|
|
|
|
|
|
2018-04-11 16:47:36 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with success (value `true`) when
|
|
|
|
|
|
* existing messages of the target actor has been processed and the actor has been
|
|
|
|
|
|
* terminated.
|
|
|
|
|
|
*
|
|
|
|
|
|
* Useful when you need to wait for termination or compose ordered termination of several actors.
|
|
|
|
|
|
*
|
|
|
|
|
|
* If you want to invoke specialized stopping logic on your target actor instead of PoisonPill, you can pass your
|
|
|
|
|
|
* stop command as `stopMessage` parameter
|
|
|
|
|
|
*
|
|
|
|
|
|
* If the target actor isn't terminated within the timeout the [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* is completed with failure [[akka.pattern.AskTimeoutException]].
|
|
|
|
|
|
*/
|
|
|
|
|
|
def gracefulStop(target: ActorRef, timeout: java.time.Duration, stopMessage: Any): CompletionStage[java.lang.Boolean] =
|
|
|
|
|
|
scalaGracefulStop(target, timeout.asScala, stopMessage).toJava.asInstanceOf[CompletionStage[java.lang.Boolean]]
|
|
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with the success or failure of the provided Callable
|
|
|
|
|
|
* after the specified duration.
|
|
|
|
|
|
*/
|
2018-04-11 16:47:36 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.12")
|
2016-01-21 16:37:26 +01:00
|
|
|
|
def after[T](duration: FiniteDuration, scheduler: Scheduler, context: ExecutionContext, value: Callable[CompletionStage[T]]): CompletionStage[T] =
|
|
|
|
|
|
afterCompletionStage(duration, scheduler)(value.call())(context)
|
|
|
|
|
|
|
2018-04-11 16:47:36 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with the success or failure of the provided Callable
|
|
|
|
|
|
* after the specified duration.
|
|
|
|
|
|
*/
|
|
|
|
|
|
def after[T](duration: java.time.Duration, scheduler: Scheduler, context: ExecutionContext, value: Callable[CompletionStage[T]]): CompletionStage[T] =
|
|
|
|
|
|
afterCompletionStage(duration.asScala, scheduler)(value.call())(context)
|
|
|
|
|
|
|
2016-01-21 16:37:26 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with the success or failure of the provided value
|
|
|
|
|
|
* after the specified duration.
|
|
|
|
|
|
*/
|
2018-04-11 16:47:36 +02:00
|
|
|
|
@deprecated("Use the overloaded one which accepts java.time.Duration instead.", since = "2.5.12")
|
2016-01-21 16:37:26 +01:00
|
|
|
|
def after[T](duration: FiniteDuration, scheduler: Scheduler, context: ExecutionContext, value: CompletionStage[T]): CompletionStage[T] =
|
|
|
|
|
|
afterCompletionStage(duration, scheduler)(value)(context)
|
2018-03-26 14:56:20 +03:00
|
|
|
|
|
2018-04-11 16:47:36 +02:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns a [[java.util.concurrent.CompletionStage]] that will be completed with the success or failure of the provided value
|
|
|
|
|
|
* after the specified duration.
|
|
|
|
|
|
*/
|
|
|
|
|
|
def after[T](duration: java.time.Duration, scheduler: Scheduler, context: ExecutionContext, value: CompletionStage[T]): CompletionStage[T] =
|
|
|
|
|
|
afterCompletionStage(duration.asScala, scheduler)(value)(context)
|
|
|
|
|
|
|
2018-03-26 14:56:20 +03:00
|
|
|
|
/**
|
|
|
|
|
|
* Returns an internally retrying [[java.util.concurrent.CompletionStage]]
|
|
|
|
|
|
* The first attempt will be made immediately, and each subsequent attempt will be made after 'delay'.
|
|
|
|
|
|
* A scheduler (eg context.system.scheduler) must be provided to delay each retry
|
2018-06-09 17:42:56 +09:00
|
|
|
|
* If attempts are exhausted the returned completion operator is simply the result of invoking attempt.
|
2018-03-26 14:56:20 +03:00
|
|
|
|
* Note that the attempt function will be invoked on the given execution context for subsequent tries
|
|
|
|
|
|
* and therefore must be thread safe (not touch unsafe mutable state).
|
|
|
|
|
|
*/
|
2018-04-11 16:47:36 +02:00
|
|
|
|
def retry[T](attempt: Callable[CompletionStage[T]], attempts: Int, delay: java.time.Duration, scheduler: Scheduler, ec: ExecutionContext): CompletionStage[T] =
|
|
|
|
|
|
scalaRetry(() ⇒ attempt.call().toScala, attempts, delay.asScala)(ec, scheduler).toJava
|
2016-01-21 16:37:26 +01:00
|
|
|
|
}
|