using System.Collections.Generic;
using System.Threading.Tasks;
public sealed class MyStopMessage{
public static readonly MyStopMessage Instance = new MyStopMessage();
private MyStopMessage(){}
public sealed class StartWork{
public int WorkCount {get;}
public StartWork(int workCount){
public sealed class ActorA : ReceiveActor{
private IActorRef _actorB;
private readonly ILoggingAdapter _log = Context.GetLogger();
Receive<StartWork>(w => {
foreach(var i in Enumerable.Range(0, w.WorkCount)){
ReceiveAsync<MyStopMessage>(async _ => {
_log.Info("Begin shutdown");
await _actorB.GracefulStop(TimeSpan.FromSeconds(10), _);
protected override void PreStart(){
_actorB = Context.ActorOf(Props.Create(() => new ActorB()), "b");
public sealed class ActorB : ReceiveActor{
private IActorRef _actorC;
private IActorRef _actorD;
private readonly ILoggingAdapter _log = Context.GetLogger();
ReceiveAsync<MyStopMessage>(async _ => {
_log.Info("Begin shutdown");
var stopC = _actorC.GracefulStop(TimeSpan.FromSeconds(10));
var stopD = _actorD.GracefulStop(TimeSpan.FromSeconds(10));
var bothStopped = Task.WhenAll(stopC, stopD);
protected override void PreStart(){
var workerProps = Props.Create(() => new WorkerActor());
_actorC = Context.ActorOf(workerProps, "c");
_actorD = Context.ActorOf(workerProps, "d");
public sealed class WorkerActor : ReceiveActor {
private readonly ILoggingAdapter _log = Context.GetLogger();
ReceiveAsync<int>(async i => {
_log.Info("Received {0}", i);
public static async Task Main()
var actorSystem = ActorSystem.Create("test");
var actorA = actorSystem.ActorOf(Props.Create(() => new ActorA()), "a");
actorA.Tell(new StartWork(100));
await actorA.GracefulStop(TimeSpan.FromSeconds(12), MyStopMessage.Instance);