it.unibo.alchemist.core.Engine Maven / Gradle / Ivy
/*
* Copyright (C) 2010-2023, Danilo Pianini and contributors
* listed, for each module, in the respective subproject's build.gradle.kts file.
*
* This file is part of Alchemist, and is distributed under the terms of the
* GNU General Public License, with a linking exception,
* as described in the file LICENSE in the Alchemist distribution's top directory.
*/
package it.unibo.alchemist.core;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Sets;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import it.unibo.alchemist.boundary.OutputMonitor;
import it.unibo.alchemist.model.Actionable;
import it.unibo.alchemist.model.Context;
import it.unibo.alchemist.model.Dependency;
import it.unibo.alchemist.model.Environment;
import it.unibo.alchemist.model.Neighborhood;
import it.unibo.alchemist.model.Node;
import it.unibo.alchemist.model.Position;
import it.unibo.alchemist.model.Reaction;
import it.unibo.alchemist.model.Time;
import org.jooq.lambda.fi.lang.CheckedRunnable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.ArrayDeque;
import java.util.Arrays;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Optional;
import java.util.Queue;
import java.util.Set;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.function.BooleanSupplier;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Stream;
import static it.unibo.alchemist.core.Status.PAUSED;
import static it.unibo.alchemist.core.Status.RUNNING;
import static it.unibo.alchemist.core.Status.TERMINATED;
/**
* This class implements a simulation. It offers a wide number of static
* factories to ease the creation process.
*
* @param concentration type
* @param {@link Position} type
*/
public class Engine> implements Simulation {
/**
* Logger.
*/
protected static final Logger LOGGER = LoggerFactory.getLogger(Engine.class);
private final Lock statusLock = new ReentrantLock();
private final ImmutableMap statusLocks = Arrays.stream(Status.values())
.collect(ImmutableMap.toImmutableMap(Function.identity(), it -> new SynchBox()));
private final BlockingQueue commands = new LinkedBlockingQueue<>();
private final Queue afterExecutionUpdates = new ArrayDeque<>();
private final Environment environment;
private final DependencyGraph dependencyGraph;
private final Scheduler scheduler;
private final Time finalTime;
private final List> monitors = new CopyOnWriteArrayList<>();
private final long finalStep;
private volatile Status status = Status.INIT;
private Optional error = Optional.empty();
private Time currentTime = Time.ZERO;
private long currentStep;
private Thread simulationThread;
/**
* Builds a simulation for a given environment. By default, it uses a
* DependencyGraph and an IndexedPriorityQueue internally. If you want to
* use your own implementations of {@link DependencyGraph} and
* {@link Scheduler} interfaces, don't use this constructor.
*
* @param e the environment at the initial time
*/
public Engine(final Environment e) {
this(e, Time.INFINITY);
}
/**
* Builds a simulation for a given environment. By default, it uses a
* DependencyGraph and an IndexedPriorityQueue internally. If you want to
* use your own implementations of {@link DependencyGraph} and
* {@link Scheduler} interfaces, don't use this constructor.
*
*
* @param e the environment at the initial time
* @param maxSteps the maximum number of steps to take
*/
public Engine(final Environment e, final long maxSteps) {
this(e, maxSteps, Time.INFINITY);
}
/**
* Builds a simulation for a given environment. By default, it uses a
* DependencyGraph and an IndexedPriorityQueue internally. If you want to
* use your own implementations of {@link DependencyGraph} and
* {@link Scheduler} interfaces, don't use this constructor.
*
* @param e the environment at the initial time
* @param maxSteps the maximum number of steps to take
* @param t the maximum time to reach
*/
@SuppressFBWarnings(
value = {"EI_EXPOSE_REP", "MC_OVERRIDABLE_METHOD_CALL_IN_CONSTRUCTOR"},
justification = "The environment is stored intentionally, and this class is final"
)
public Engine(final Environment e, final long maxSteps, final Time t) {
this(e, maxSteps, t, new ArrayIndexedPriorityQueue<>());
}
/**
* Builds a simulation for a given environment. By default, it uses a
* DependencyGraph and an IndexedPriorityBatchQueue internally. If you want to
* use your own implementations of {@link DependencyGraph} and
* {@link Scheduler} interfaces, don't use this constructor.
*
* @param e t
* he environment at the initial time
* @param maxSteps the maximum number of steps to take
* @param t the maximum time to reach
* @param scheduler the scheduler implementation to be used
*/
@SuppressFBWarnings(
value = {"EI_EXPOSE_REP2", "MC_OVERRIDABLE_METHOD_CALL_IN_CONSTRUCTOR"},
justification = "Environment and scheduler are not clonable, setSimulation is not final")
public Engine(final Environment e, final long maxSteps, final Time t, final Scheduler scheduler) {
LOGGER.trace("Engine created");
environment = e;
environment.setSimulation(this);
dependencyGraph = new JGraphTDependencyGraph<>(environment);
this.scheduler = scheduler;
this.finalStep = maxSteps;
this.finalTime = t;
}
/**
* Builds a simulation for a given environment. By default, it uses a
* DependencyGraph and an IndexedPriorityQueue internally. If you want to
* use your own implementations of {@link DependencyGraph} and
* {@link Scheduler} interfaces, don't use this constructor.
*
* @param e the environment at the initial time
* @param t the maximum time to reach
*/
public Engine(final Environment e, final Time t) {
this(e, Long.MAX_VALUE, t);
}
/**
* @param op the OutputMonitor to add
*/
@Override
public void addOutputMonitor(final OutputMonitor op) {
monitors.add(op);
}
private void checkCaller() {
if (this.getClass() != BatchEngine.class && !Thread.currentThread().equals(simulationThread)) {
throw new IllegalMonitorStateException("This method must get called from the simulation thread.");
}
}
private R doOnStatus(final Supplier fun) {
try {
statusLock.lock();
return fun.get();
} finally {
statusLock.unlock();
}
}
private void doOnStatus(final Runnable fun) {
doOnStatus(() -> {
fun.run();
return 0;
});
}
/**
* Perform a single step of the simulation.
*/
protected void doStep() {
final Actionable nextEvent = scheduler.getNext();
if (nextEvent == null) {
this.newStatus(TERMINATED);
LOGGER.info("No more reactions.");
} else {
final Time scheduledTime = nextEvent.getTau();
if (scheduledTime.compareTo(getTime()) < 0) {
throw new IllegalStateException(
nextEvent + " is scheduled in the past at time " + scheduledTime
+ ", current time is " + getTime()
+ ". Problem occurred at step " + getStep()
);
}
setCurrentTime(scheduledTime);
if (nextEvent.canExecute()) {
/*
* This must be taken before execution, because the reaction
* might remove itself (or its node) from the environment.
*/
nextEvent.getConditions().forEach(it.unibo.alchemist.model.Condition::reactionReady);
nextEvent.execute();
Set> toUpdate = dependencyGraph.outboundDependencies(nextEvent);
if (!afterExecutionUpdates.isEmpty()) {
afterExecutionUpdates.forEach(Update::performChanges);
afterExecutionUpdates.clear();
toUpdate = Sets.union(toUpdate, dependencyGraph.outboundDependencies(nextEvent));
}
toUpdate.forEach(this::updateReaction);
}
nextEvent.update(getTime(), true, environment);
scheduler.updateReaction(nextEvent);
for (final OutputMonitor monitor : monitors) {
monitor.stepDone(environment, nextEvent, getTime(), getStep());
}
}
if (environment.isTerminated()) {
newStatus(TERMINATED);
LOGGER.info("Termination condition reached.");
}
final var newStep = getStep() + 1;
setCurrentStep(newStep);
}
private void finalizeConstructor() {
this.environment.getGlobalReactions().forEach(this::reactionAdded);
for (final Node n : environment) {
for (final Reaction r : n.getReactions()) {
scheduleReaction(r);
}
}
}
/**
* @return environment
*/
@Override
@SuppressFBWarnings("EI_EXPOSE_REP")
public Environment getEnvironment() {
return environment;
}
/**
* @return error
*/
@Override
public Optional getError() {
return error;
}
/**
* @return final step
*/
@Override
public long getFinalStep() {
return finalStep;
}
/**
* @return final time
*/
@Override
public Time getFinalTime() {
return finalTime;
}
/**
* @return status
*/
@Override
public Status getStatus() {
return status;
}
/**
* thread-safe.
*
* @return step
*/
@Override
public synchronized long getStep() {
return currentStep;
}
/**
* thread-safe.
*
* @return time
*/
@Override
public synchronized Time getTime() {
return currentTime;
}
/**
* @param step the number of steps to execute
*/
@Override
public void goToStep(final long step) {
pauseWhen(() -> getStep() >= step);
}
/**
* @param t the target time
*/
@Override
public void goToTime(final Time t) {
pauseWhen(() -> getTime().compareTo(t) >= 0);
}
private void idleProcessSingleCommand() throws Throwable {
CheckedRunnable nextCommand = null;
// This is for spurious wakeups. Blame Java.
while (nextCommand == null) {
try {
nextCommand = commands.take();
processCommand(nextCommand);
} catch (InterruptedException e) {
LOGGER.debug("Look! A spurious wakeup! :-)");
}
}
}
/**
* @param node the node
* @param n the second node
*/
@Override
public void neighborAdded(final Node node, final Node n) {
checkCaller();
afterExecutionUpdates.add(new NeigborAdded(node, n));
}
/**
* @param node the node
* @param n the second node
*/
@Override
public void neighborRemoved(final Node node, final Node n) {
checkCaller();
afterExecutionUpdates.add(new NeigborRemoved(node, n));
}
/**
* Sets new status.
*
* @param next next status
*/
protected void newStatus(final Status next) {
schedule(() -> doOnStatus(() -> {
if (next.isReachableFrom(status)) {
status = next;
lockForStatus(next).releaseAll();
}
}));
}
/**
* @param node the freshly added node
*/
@Override
public void nodeAdded(final Node node) {
checkCaller();
afterExecutionUpdates.add(new NodeAddition(node));
}
/**
* @param node the node
*/
@Override
public void nodeMoved(final Node node) {
checkCaller();
afterExecutionUpdates.add(new Movement(node));
}
/**
* @param node the freshly removed node
* @param oldNeighborhood the neighborhood of the node as it was before it was removed
* (used to calculate reverse dependencies)
*/
@Override
public void nodeRemoved(final Node node, final Neighborhood oldNeighborhood) {
checkCaller();
afterExecutionUpdates.add(new NodeRemoval(node));
}
/**
* pause.
*/
@Override
public void pause() {
newStatus(PAUSED);
}
/**
* play.
*/
@Override
public void play() {
newStatus(RUNNING);
}
/**
* @param reactionToAdd the reaction to add
*/
@Override
public void reactionAdded(final Actionable reactionToAdd) {
reactionChanged(new ReactionAddition(reactionToAdd));
}
/**
* @param reactionToRemove the reaction to remove
*/
@Override
public void reactionRemoved(final Actionable reactionToRemove) {
reactionChanged(new ReactionRemoval(reactionToRemove));
}
private void reactionChanged(final UpdateOnReaction update) {
checkCaller();
afterExecutionUpdates.add(update);
}
private Stream extends Actionable> reactionsToUpdateAfterExecution() {
return afterExecutionUpdates.stream()
.flatMap(Update::getReactionsToUpdate)
.distinct();
}
private void processCommand(final CheckedRunnable command) throws Throwable {
command.run();
// Update all reactions before applying dependency graph updates
final Set> updated = new LinkedHashSet<>();
reactionsToUpdateAfterExecution().forEach(r -> {
updated.add(r);
updateReaction(r);
});
// Now update the dependency graph as needed
afterExecutionUpdates.forEach(Update::performChanges);
afterExecutionUpdates.clear();
// Now update the new reactions
reactionsToUpdateAfterExecution().forEach(r -> {
if (!updated.contains(r)) {
updateReaction(r);
}
});
}
/**
* @param op the OutputMonitor to add
*/
@Override
public void removeOutputMonitor(final OutputMonitor op) {
monitors.remove(op);
}
/**
* run simulation.
*/
@Override
public void run() {
synchronized (environment) {
try {
LOGGER.info("Starting engine {} with scheduler {}", this.getClass(), scheduler.getClass());
simulationThread = Thread.currentThread();
finalizeConstructor();
status = Status.READY;
final long currentThread = Thread.currentThread().getId();
LOGGER.trace("Thread {} started running.", currentThread);
for (final OutputMonitor m : monitors) {
m.initialized(environment);
}
while (status.equals(Status.READY)) {
idleProcessSingleCommand();
}
while (status != TERMINATED && getStep() < finalStep && getTime().compareTo(finalTime) < 0) {
while (!commands.isEmpty()) {
processCommand(commands.poll());
}
if (status.equals(RUNNING)) {
doStep();
}
while (status.equals(PAUSED)) {
idleProcessSingleCommand();
}
}
} catch (Throwable e) { // NOPMD: forced by CheckedRunnable
error = Optional.of(e);
LOGGER.error("The simulation engine crashed.", e);
} finally {
status = TERMINATED;
commands.clear();
try {
for (final OutputMonitor m : monitors) {
m.finished(environment, getTime(), getStep());
}
} catch (Throwable e) { //NOPMD: we need to catch everything
error.ifPresentOrElse(
error -> error.addSuppressed(e),
() -> error = Optional.of(e)
);
}
afterRun();
}
}
}
/**
* Override this to execute something after simulation run.
*/
protected void afterRun() {
// do nothing, leave for override...
}
/**
* @param condition condition
*/
private void pauseWhen(final BooleanSupplier condition) {
addOutputMonitor(new OutputMonitor<>() {
@Override
public void initialized(@Nonnull final Environment environment) {
if (condition.getAsBoolean()) {
monitors.remove(this);
pause();
}
}
@Override
public void stepDone(
@Nonnull final Environment environment,
@Nullable final Actionable reaction,
@Nonnull final Time time,
final long step
) {
initialized(environment);
}
});
}
/**
* @param runnable the runnable to execute
*/
@Override
public void schedule(final CheckedRunnable runnable) {
if (getStatus().equals(TERMINATED)) {
throw new IllegalStateException("This simulation is terminated and can not get resumed.");
}
commands.add(runnable);
}
private void scheduleReaction(final Actionable reaction) {
dependencyGraph.createDependencies(reaction);
reaction.initializationComplete(getTime(), environment);
scheduler.addReaction(reaction);
}
/**
* terminate.
*/
@Override
public void terminate() {
newStatus(TERMINATED);
}
/**
* @return string representation of the engine
*/
@Override
public String toString() {
return getClass().getSimpleName() + " t: " + getTime() + ", s: " + getStep();
}
/**
* update reaction.
*
* @param r reaction to be updated
*/
protected void updateReaction(final Actionable r) {
final Time t = r.getTau();
r.update(getTime(), false, environment);
if (!r.getTau().equals(t)) {
scheduler.updateReaction(r);
}
}
private SynchBox lockForStatus(final Status status) {
final var statusLock = statusLocks.get(status);
if (statusLock == null) {
throw new IllegalStateException(
"Inconsistent state, the Alchemist engine tried to synchronize on a non-existing lock"
+ "searching for status: " + status + ", available locks: " + statusLocks
);
}
return statusLock;
}
/**
* @param next The {@link Status} the simulation should reach before returning from this method
* @param timeout The maximum lapse of time the caller wants to wait before being resumed
* @param tu The {@link TimeUnit} used to define "timeout"
* @return status
*/
@Override
public Status waitFor(final Status next, final long timeout, final TimeUnit tu) {
return lockForStatus(next).waitFor(next, timeout, tu);
}
/**
* Returns output monitors.
*
* @return output monitors
*/
@Override
public List> getOutputMonitors() {
return ImmutableList.copyOf(monitors);
}
/**
* Returns after execution updates.
*
* @return after execution updates
*/
protected Queue getAfterExecutionUpdates() {
return afterExecutionUpdates;
}
/**
* Returns dependency graph.
*
* @return dependency graph
*/
protected DependencyGraph getDependencyGraph() {
return dependencyGraph;
}
/**
* Returns scheduler.
*
* @return scheduler
*/
protected Scheduler getScheduler() {
return scheduler;
}
/**
* Returns status lock.
*
* @return status locks
*/
protected ImmutableMap getStatusLocks() {
return statusLocks;
}
/**
* Returns monitors.
*
* @return monitors
*/
protected List> getMonitors() {
return monitors;
}
/**
* thread safe. Updates current time
*
* @param currentTime new current time
*/
protected synchronized void setCurrentTime(final Time currentTime) {
this.currentTime = currentTime;
}
/**
* thread safe. Updates current step.
*
* @param currentStep new current step
*/
protected synchronized void setCurrentStep(final long currentStep) {
this.currentStep = currentStep;
}
/**
* Class representing an update.
*/
// CHECKSTYLE: FinalClassCheck OFF
protected class Update {
/**
* Perform changes.
*/
public void performChanges() {
// override me
}
/**
* @return reactions to update
*/
public Stream extends Actionable> getReactionsToUpdate() {
return Stream.empty();
}
}
private final class Movement extends Update {
private final @Nonnull Node sourceNode;
private Movement(final @Nonnull Node sourceNode) {
this.sourceNode = sourceNode;
}
@Override
public Stream extends Actionable> getReactionsToUpdate() {
return getReactionsRelatedTo(this.sourceNode, environment.getNeighborhood(this.sourceNode))
.filter(it ->
it.getInboundDependencies().stream()
.anyMatch(dependency -> dependency.dependsOn(Dependency.MOVEMENT))
);
}
private Stream extends Actionable> getReactionsRelatedTo(
final Node source,
final Neighborhood neighborhood
) {
return Stream.of(
source.getReactions().stream(),
neighborhood.getNeighbors().stream()
.flatMap(node -> node.getReactions().stream())
.filter(it -> it.getInputContext() == Context.NEIGHBORHOOD),
dependencyGraph.globalInputContextReactions().stream())
.reduce(Stream.empty(), Stream::concat);
}
}
private class UpdateOnNode extends Update {
private final Function, Update> reactionLevelOperation;
private final Node sourceNode;
private UpdateOnNode(final Node sourceNode, final Function, Update> reactionLevelOperation) {
this.sourceNode = sourceNode;
this.reactionLevelOperation = reactionLevelOperation;
}
@Override
public final void performChanges() {
this.sourceNode.getReactions().stream()
.map(reactionLevelOperation)
.forEach(Update::performChanges);
}
}
private final class NodeRemoval extends UpdateOnNode {
private NodeRemoval(final Node sourceNode) {
super(sourceNode, ReactionRemoval::new);
}
}
private final class NodeAddition extends UpdateOnNode {
private NodeAddition(final Node sourceNode) {
super(sourceNode, ReactionAddition::new);
}
}
private abstract class UpdateOnReaction extends Update {
private final Actionable sourceActionable;
private UpdateOnReaction(final Actionable sourceActionable) {
this.sourceActionable = sourceActionable;
}
@Override
public final Stream extends Actionable> getReactionsToUpdate() {
return Stream.of(sourceActionable);
}
protected Actionable getSourceReaction() {
return sourceActionable;
}
}
private final class ReactionRemoval extends UpdateOnReaction {
private ReactionRemoval(final Actionable source) {
super(source);
}
@Override
public void performChanges() {
dependencyGraph.removeDependencies(getSourceReaction());
scheduler.removeReaction(getSourceReaction());
}
}
private final class ReactionAddition extends UpdateOnReaction {
private ReactionAddition(final Actionable source) {
super(source);
}
@Override
public void performChanges() {
Engine.this.scheduleReaction(getSourceReaction());
}
}
private class NeighborhoodChanged extends Update {
private final Node sourceNode;
private final Node targetNode;
private NeighborhoodChanged(final Node sourceNode, final Node targetNode) {
this.sourceNode = sourceNode;
this.targetNode = targetNode;
}
public Node getTargetNode() {
return targetNode;
}
public Node getSourceNode() {
return sourceNode;
}
@Override
public Stream extends Actionable> getReactionsToUpdate() {
return Stream.of(
Stream.concat(
// source, target, and all their neighbors are candidates.
Stream.of(sourceNode, targetNode),
Stream.of(environment.getNeighborhood(sourceNode), environment.getNeighborhood(getTargetNode()))
.flatMap(it -> it.getNeighbors().stream()))
.distinct()
.flatMap(it -> it.getReactions().stream())
.filter(it -> it.getInputContext() == Context.NEIGHBORHOOD),
// Global reactions
dependencyGraph.globalInputContextReactions().stream())
.reduce(Stream.empty(), Stream::concat);
}
}
private final class NeigborAdded extends NeighborhoodChanged {
private NeigborAdded(final Node source, final Node target) {
super(source, target);
}
@Override
public void performChanges() {
dependencyGraph.addNeighbor(getSourceNode(), getTargetNode());
}
}
private final class NeigborRemoved extends NeighborhoodChanged {
private NeigborRemoved(final Node source, final Node target) {
super(source, target);
}
@Override
public void performChanges() {
dependencyGraph.removeNeighbor(getSourceNode(), getTargetNode());
}
}
private final class SynchBox {
private final AtomicInteger queueLength = new AtomicInteger();
private final Condition statusReached = statusLock.newCondition();
private final Condition allReleased = statusLock.newCondition();
public Status waitFor(final Status next, final long timeout, final TimeUnit tu) {
return doOnStatus(() -> {
boolean notTimedOut = true;
while (notTimedOut && next != status && next.isReachableFrom(status)) {
try {
queueLength.getAndIncrement();
notTimedOut = statusReached.await(timeout, tu);
queueLength.getAndDecrement();
} catch (InterruptedException e) {
LOGGER.info("Spurious wakeup?", e);
}
}
if (queueLength.get() == 0) {
allReleased.signal();
}
return status;
});
}
public void releaseAll() {
doOnStatus(() -> {
while (queueLength.get() != 0) {
statusReached.signalAll();
allReleased.awaitUninterruptibly();
}
});
}
}
}