package docs.cluster; import java.util.ArrayList; import java.util.List; import java.util.concurrent.TimeUnit; import docs.cluster.StatsMessages.JobFailed; import docs.cluster.StatsMessages.StatsResult; import scala.concurrent.duration.Duration; import akka.actor.ActorRef; import akka.actor.ReceiveTimeout; import akka.actor.AbstractActor; //#aggregator public class StatsAggregator extends AbstractActor { final int expectedResults; final ActorRef replyTo; final List results = new ArrayList(); public StatsAggregator(int expectedResults, ActorRef replyTo) { this.expectedResults = expectedResults; this.replyTo = replyTo; } @Override public void preStart() { getContext().setReceiveTimeout(Duration.create(3, TimeUnit.SECONDS)); } @Override public Receive createReceive() { return receiveBuilder() .match(Integer.class, wordCount -> { results.add(wordCount); if (results.size() == expectedResults) { int sum = 0; for (int c : results) { sum += c; } double meanWordLength = ((double) sum) / results.size(); replyTo.tell(new StatsResult(meanWordLength), self()); getContext().stop(self()); } }) .match(ReceiveTimeout.class, x -> { replyTo.tell(new JobFailed("Service unavailable, try again later"), self()); getContext().stop(self()); }) .build(); } } //#aggregator