Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
it.unibo.alchemist.model.implementations.reactions.SAPEREReaction Maven / Gradle / Ivy
/*
* Copyright (C) 2010-2019, Danilo Pianini and contributors listed in the main project's alchemist/build.gradle 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.model.implementations.reactions;
import it.unibo.alchemist.expressions.implementations.NumTreeNode;
import it.unibo.alchemist.expressions.interfaces.ITreeNode;
import it.unibo.alchemist.model.interfaces.Dependency;
import org.apache.commons.math3.random.RandomGenerator;
import it.unibo.alchemist.model.implementations.actions.LsaStandardAction;
import it.unibo.alchemist.model.implementations.molecules.LsaMolecule;
import it.unibo.alchemist.model.implementations.timedistributions.SAPERETimeDistribution;
import it.unibo.alchemist.model.interfaces.Context;
import it.unibo.alchemist.model.interfaces.Action;
import it.unibo.alchemist.model.interfaces.Condition;
import it.unibo.alchemist.model.interfaces.Environment;
import it.unibo.alchemist.model.interfaces.ILsaAction;
import it.unibo.alchemist.model.interfaces.ILsaCondition;
import it.unibo.alchemist.model.interfaces.ILsaMolecule;
import it.unibo.alchemist.model.interfaces.ILsaNode;
import it.unibo.alchemist.model.interfaces.Node;
import it.unibo.alchemist.model.interfaces.Position;
import it.unibo.alchemist.model.interfaces.Reaction;
import it.unibo.alchemist.model.interfaces.Time;
import it.unibo.alchemist.model.interfaces.TimeDistribution;
import org.danilopianini.lang.HashString;
import org.danilopianini.util.ArrayListSet;
import org.danilopianini.util.ListSet;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Objects;
/**
* This class realizes a reaction with Lsa concentrations.
*
*
*/
@SuppressWarnings("unchecked")
public final class SAPEREReaction extends AbstractReaction> {
private static final long serialVersionUID = 1L;
private final Environment, ?> environment;
@SuppressFBWarnings(
value = "SE_BAD_FIELD",
justification = "All provided RandomGenerator implementations are actually Serializable"
)
private final RandomGenerator rng;
private final SAPERETimeDistribution timeDistribution;
private boolean emptyExecution;
private boolean modifiesOnlyLocally = true;
private List>> possibleMatches = new ArrayList<>(0);
private List>> possibleRemove = new ArrayList<>(0);
private List propensities = new ArrayList<>(0);
private double totalPropensity;
private List validNodes = new ArrayList<>(0);
/**
* This method screens the lsaMolecule list, deleting all molecule which can
* be covered from another one more generic.
*
* @param moleculeList
* List of lsaMolecule to screen
*/
private static void screen(final List moleculeList) {
/*
* PHASE 1: generalize the list
*/
for (int i = 0; i < moleculeList.size(); i++) { // NOPMD: this loop has side effects
moleculeList.add(((ILsaMolecule) moleculeList.remove(0)).generalize());
}
/*
* PHASE2: compare one-by-one
*/
for (int i = moleculeList.size() - 1; i > 0; i--) {
for (int p = i - 1; p >= 0; p--) {
final ILsaMolecule m1 = (ILsaMolecule) moleculeList.get(i);
final ILsaMolecule m2 = (ILsaMolecule) moleculeList.get(p);
if (m2.equals(m1) || m2.moreGenericOf(m1)) {
moleculeList.remove(i);
i--;
} else if (m1.moreGenericOf(m2)) {
moleculeList.remove(p);
i--;
}
}
}
}
/**
* @param environment
* the current environment
* @param node
* the current node
* @param randomGenerator
* the random engine to use
* @param timeDistribution
* Time Distribution
*/
public SAPEREReaction(
final Environment, ?> environment,
final ILsaNode node,
final RandomGenerator randomGenerator,
final TimeDistribution> timeDistribution
) {
super(node, timeDistribution);
if (getTimeDistribution() instanceof SAPERETimeDistribution) {
this.timeDistribution = (SAPERETimeDistribution) getTimeDistribution();
} else {
this.timeDistribution = null;
}
rng = randomGenerator;
this.environment = environment;
}
@Override
public Reaction> cloneOnNewNode(final Node> node, final Time currentTime) {
final var timeDistributionClone = timeDistribution.cloneOnNewNode(node, currentTime);
final SAPEREReaction res = new SAPEREReaction(environment, (ILsaNode) node, rng, timeDistributionClone);
final ArrayList>> c = new ArrayList<>();
for (final Condition> cond : getConditions()) {
c.add(cond.cloneCondition(node, res));
}
final ArrayList>> a = new ArrayList<>();
for (final Action> act : getActions()) {
a.add(act.cloneAction(node, res));
}
res.setActions(a);
res.setConditions(c);
return res;
}
/**
* @return the inner {@link Action} list, cast
*/
protected List getSAPEREActions() {
return (List) (List extends Action>>) getActions();
}
/**
* @return the inner {@link Condition} list, cast
*/
protected List getSAPEREConditions() {
return (List) (List extends Condition>>) getConditions();
}
@Override
public void execute() {
if (possibleMatches.isEmpty()) {
for (final ILsaAction a : getSAPEREActions()) {
a.setExecutionContext(null, validNodes);
a.execute();
}
return;
}
final Position> nodePosCache = modifiesOnlyLocally ? environment.getPosition(getNode()) : null;
final List extends ILsaMolecule> localContentCache = modifiesOnlyLocally
? new ArrayList<>(getLsaNode().getLsaSpace())
: null;
Map> matches = null;
Map> toRemove = null;
/*
* If there is infinite propensity, the last match added is the one to
* choose, since it is the one which generated the "infinity" value.
*/
if (totalPropensity == Double.POSITIVE_INFINITY) {
final int index = possibleMatches.size() - 1;
matches = possibleMatches.get(index);
toRemove = possibleRemove.get(index);
} else if (numericRate()) {
/*
* If the rate is numeric, the choice is just random
*/
final int index = Math.abs(rng.nextInt()) % possibleMatches.size();
matches = possibleMatches.get(index);
toRemove = possibleRemove.get(index);
} else {
/*
* Otherwise, the matches must be chosen randomly using their
* propensities
*/
final double index = rng.nextDouble() * totalPropensity;
double sum = 0;
for (int i = 0; matches == null; i++) {
sum += propensities.get(i);
if (sum > index) {
matches = possibleMatches.get(i);
toRemove = possibleRemove.get(i);
}
}
}
/*
* The matched LSAs must be removed from the local space, if no action
* added them back.
*/
for (final Entry> entry : toRemove.entrySet()) {
final ILsaNode n = entry.getKey();
for (final ILsaMolecule m : entry.getValue()) {
n.removeConcentration(m);
}
}
/*
* #T Must be loaded by the reaction, which is the only structure aware
* of the time. Other special values (#NEIG, #O, #D) will be allocated
* inside the actions.
*/
matches.put(LsaMolecule.SYN_T, new NumTreeNode(getTau().toDouble()));
for (final ILsaAction a : getSAPEREActions()) {
a.setExecutionContext(matches, validNodes);
a.execute();
}
/*
* Empty action optimization
*/
if (modifiesOnlyLocally) {
final ILsaNode n = getLsaNode();
if (Objects.requireNonNull(nodePosCache).equals(environment.getPosition(getNode()))) {
final List extends ILsaMolecule> contents = n.getLsaSpace();
if (contents.size() == Objects.requireNonNull(localContentCache).size()) {
emptyExecution = localContentCache.containsAll(contents);
}
}
}
}
/**
* @return the current environment
*/
protected Environment, ?> getEnvironment() {
return environment;
}
/**
* @return the local {@link Node} as {@link ILsaNode}
*/
protected ILsaNode getLsaNode() {
return (ILsaNode) super.getNode();
}
/**
* @return the list of all possible matches
*/
protected List>> getPossibleMatches() {
return possibleMatches;
}
/**
* @return the list of molecules which would be removed for each node if the
* corresponding match would be chosen
*/
protected List>> getPossibleRemove() {
return possibleRemove;
}
/**
* @return the list of the propensities computed for each match combination
*/
protected List getPropensities() {
return propensities;
}
@Override
protected void updateInternalStatus(
final Time currentTime,
final boolean hasBeenExecuted,
final Environment, ?> environment
) {
if (emptyExecution) {
emptyExecution = false;
totalPropensity = 0;
} else {
/*
* Valid nodes must be re-initialized, as per issue #
*/
final Collection extends Node>> neighs =
this.environment.getNeighborhood(getNode()).getNeighbors();
validNodes = new ArrayList<>(neighs.size());
for (final Node> neigh: neighs) {
validNodes.add((ILsaNode) neigh);
}
if (getConditions().isEmpty()) {
totalPropensity = getTimeDistribution().getRate();
} else {
totalPropensity = 0d;
possibleMatches = new ArrayList<>();
propensities = new ArrayList<>();
possibleRemove = new ArrayList<>();
/*
* Apply all the conditions as filters
*/
for (final ILsaCondition cond : getSAPEREConditions()) {
if (!cond.filter(possibleMatches, validNodes, possibleRemove)) {
/*
* It is supposed that a condition fails if it must put null
* in the filter lists, so null values are not expected.
*/
return;
}
}
if (numericRate()) {
totalPropensity = possibleMatches.size() * getTimeDistribution().getRate();
} else {
/*
* For each possible match, compute the propensity
*/
for (final Map> match : possibleMatches) {
timeDistribution.setMatches(match);
final double p = timeDistribution.getRate();
propensities.add(p);
totalPropensity += p;
if (totalPropensity == Double.POSITIVE_INFINITY) {
return;
}
}
}
}
}
}
private boolean numericRate() {
return timeDistribution == null || timeDistribution.isStatic();
}
@Override
public double getRate() {
return totalPropensity;
}
@Override
public String getRateAsString() {
return numericRate() ? Double.toString(getTimeDistribution().getRate()) : timeDistribution.getRateEquation().toString();
}
/**
* @return the aggregated propensity of all the possible instances of this
* reaction
*/
protected double getTotalPropensity() {
return totalPropensity;
}
/**
* @return the list of nodes which are valid for this reaction
*/
protected List getValidNodes() {
return validNodes;
}
@Override
public void setActions(final List>> actions) {
setConditionsAndActions(getConditions(), actions);
}
@Override
public void setConditions(final List>> conditions) {
setConditionsAndActions(conditions, getActions());
}
private void setConditionsAndActions(
final List extends Condition>> c,
final List extends Action>> a
) {
super.setConditions((List>>) c);
super.setActions((List>>) a);
modifiesOnlyLocally = getOutputContext() == Context.LOCAL;
/*
* The following optimization only makes sense if the reaction acts
* locally. Otherwise there is no control on where the modified
* molecules will end up.
*/
final ListSet inboundDependencies = new ArrayListSet<>(getInboundDependencies());
final ListSet outboundDependencies = new ArrayListSet<>(getOutboundDependencies());
if (getInputContext() == Context.LOCAL && modifiesOnlyLocally) {
/*
* Moreover, since there is no control over the personalised agents,
* it's required to check that all the actions are the standard
* manipulations.
*/
boolean allStandard = true;
for (final Action> act : getActions()) {
if (!(act instanceof LsaStandardAction)) {
allStandard = false;
break;
}
}
if (allStandard) {
for (final Dependency m : inboundDependencies) {
/*
* For each influencing molecule:
*
* If it appears identically on both sides of the reaction,
* then it can be ignored when calculating the dependencies
*
* If there are some molecules on the left side which are
* not on the right side, they should be added (they will be
* removed)
*/
if (outboundDependencies.contains(m)) {
outboundDependencies.remove(m);
} else {
outboundDependencies.add(m);
}
}
}
}
screen(inboundDependencies);
screen(outboundDependencies);
inboundDependencies.forEach(this::addInboundDependency);
outboundDependencies.forEach(this::addOutboundDependency);
}
/**
* @param pm
* the list of all possible matches
*/
protected void setPossibleMatches(final List>> pm) {
this.possibleMatches = pm;
}
/**
* @param possibleRemoved
* the list of molecules which would be removed for each node if
* the corresponding match would be chosen
*/
protected void setPossibleRemove(final List>> possibleRemoved) {
this.possibleRemove = possibleRemoved;
}
/**
* @param p
* the list of the propensities computed for each match
* combination
*/
protected void setPropensities(final List p) {
this.propensities = p;
}
/**
* @param tp
* the aggregated propensity of all the possible instances of
* this reaction
*/
protected void setTotalPropensity(final double tp) {
this.totalPropensity = tp;
}
/**
* @param valid
* the list of nodes which are valid for this reaction
*/
protected void setValidNodes(final List valid) {
this.validNodes = valid;
}
}