All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.ryos.rhino.sdk.runners.EventDispatcher Maven / Gradle / Ivy

package io.ryos.rhino.sdk.runners;

import static io.ryos.rhino.sdk.reporting.GatlingSimulationLogFormatter.GATLING_HEADLINE_TEMPLATE;

import akka.actor.ActorRef;
import akka.actor.ActorSystem;
import akka.actor.Terminated;
import akka.dispatch.OnComplete;
import akka.pattern.Patterns;
import io.ryos.rhino.sdk.Simulation;
import io.ryos.rhino.sdk.SimulationMetadata;
import io.ryos.rhino.sdk.io.InfluxDBWriter;
import io.ryos.rhino.sdk.io.SimulationLogWriter;
import io.ryos.rhino.sdk.reporting.GatlingSimulationLogFormatter;
import io.ryos.rhino.sdk.reporting.Measurement;
import io.ryos.rhino.sdk.reporting.MetricCollector;
import io.ryos.rhino.sdk.reporting.MetricCollector.EndTestEvent;
import java.time.Instant;
import java.util.Objects;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import scala.concurrent.Await;
import scala.concurrent.duration.FiniteDuration;

/**
 * Singleton event dispatcher forwards the events created by simulation callables to corresponding
 * entities like actors wnich process them.
 * 

* * @author Erhan Bagdemir * @since 1.1.0 */ public class EventDispatcher { private static final Logger LOG = LogManager.getLogger(EventDispatcher.class); private static final long TERMINATION_REQUEST_TIMEOUT = 5000L; private static final String ACTOR_SYS_NAME = "rhino-dispatcher"; private static EventDispatcher INSTANCE; private EventDispatcher(final SimulationMetadata simulationMetadata) { this.simulationMetadata = Objects.requireNonNull(simulationMetadata); this.metricCollector = system .actorOf(MetricCollector.props(simulationMetadata.getNumberOfUsers(), Instant.now(), simulationMetadata.getDuration()), MetricCollector.class.getName()); this.loggerActor = system .actorOf(SimulationLogWriter.props(simulationMetadata.getReportingURI(), simulationMetadata.getLogFormatter()), SimulationLogWriter.class.getName()); if (simulationMetadata.isEnableInflux()) { influxActor = system.actorOf(InfluxDBWriter.props(), InfluxDBWriter.class.getName()); } if (simulationMetadata.getLogFormatter() instanceof GatlingSimulationLogFormatter) { loggerActor.tell( String.format( GATLING_HEADLINE_TEMPLATE, simulationMetadata.getSimulationClass().getName(), simulationMetadata.getSimulationName(), System.currentTimeMillis(), GatlingSimulationLogFormatter.GATLING_VERSION), ActorRef.noSender()); } } /** * Simulation metadata. *

*/ private SimulationMetadata simulationMetadata; /** * Reporter actor reference is the reference to the actor which receives reporting events. *

*/ private ActorRef loggerActor; /** * Reporter actor reference to write log events to the Influx DB. *

*/ private ActorRef influxActor; /** * StdOut reporter is to write out about the test execution to the stdout. It can be considered as * heartbeat about the running test. *

*/ private ActorRef metricCollector; private ActorSystem system = ActorSystem.create(ACTOR_SYS_NAME); public static EventDispatcher getInstance() { if (INSTANCE == null) { INSTANCE = new EventDispatcher(Simulation.getData().orElseThrow()); } return INSTANCE; } public void dispatchEvents(final Measurement measurement) { try { measurement.getEvents().forEach(event -> { loggerActor.tell(event, ActorRef.noSender()); metricCollector.tell(event, ActorRef.noSender()); if (simulationMetadata.isEnableInflux()) { influxActor.tell(event, ActorRef.noSender()); } }); } finally { measurement.purge(); } } public void stop() { requestForTermination(); var terminate = system.terminate(); terminate.onComplete(new OnComplete<>() { @Override public void onComplete(final Throwable throwable, final Terminated terminated) { system = null; } }, system.dispatcher()); } private void requestForTermination() { var ask = Patterns.ask(metricCollector, new EndTestEvent(Instant.now()), TERMINATION_REQUEST_TIMEOUT); try { Await.result(ask, FiniteDuration.Inf()); } catch (Exception e) { LOG.debug(e); // expected exception is a Timeout. It is ok. } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy