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

org.snpeff.reactome.events.Reaction Maven / Gradle / Ivy

The newest version!
package org.snpeff.reactome.events;

import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;

import org.snpeff.reactome.Entity;
import org.snpeff.util.Gpr;

/**
 * A reaction
 * 
 * @author pcingola
 *
 */
public class Reaction extends Event {

	/**
	 * Reaction regulation types
	 * @author pablocingolani
	 *
	 */
	public enum RegulationType {
		NegativeRegulation, PositiveRegulation, Requirement
	};

	public static final double MAX_WEIGHT_SUM = 1.1;

	protected HashMap inputs; // Input -> Weight
	protected HashSet outputs;
	protected HashSet catalyst;
	protected HashMap regulator;

	public Reaction(int id, String name) {
		super(id, name);
		inputs = new HashMap();
		outputs = new HashSet();
		catalyst = new HashSet();
		regulator = new HashMap();
	}

	public void addCatalyst(Entity e) {
		catalyst.add(e);
	}

	public void addInput(Entity e) {
		if (e == null) return;

		if (!inputs.containsKey(e)) inputs.put(e, 1.0);
		else inputs.put(e, inputs.get(e) + 1.0);
	}

	public void addOutput(Entity e) {
		outputs.add(e);

		// Add input
		if (e.isReaction()) ((Reaction) e).addInput(this);
	}

	public void addRegulator(Entity e, RegulationType type) {
		regulator.put(e, type);
	}

	public void addRegulator(Entity e, String type) {
		regulator.put(e, RegulationType.valueOf(type));
	}

	/**
	 * Calculate entities.
	 * Make sure we don't calculate twice (keep 'doneEntities' set up to date)
	 * 
	 * @param doneEntities
	 * @return
	 */
	@Override
	public double calc(HashSet doneEntities) {

		if (!Double.isNaN(fixedOutput)) output = fixedOutput;
		else {
			if (doneEntities.contains(this)) return output; // Make sure we don't calculate twice
			doneEntities.add(this); // Keep 'entities' set up to date

			//---
			// Calculate required nodes
			//---

			// Calculate inputs
			for (Entity ein : getInputs())
				ein.calc(doneEntities);

			// Calculate catalysts 
			for (Entity ecat : catalyst)
				ecat.calc(doneEntities);

			// Calculate regulators
			for (Entity ereg : regulator.keySet())
				ereg.calc(doneEntities);

			//---
			// Add inputs
			//---

			// Aggregated input
			double in = 0;
			for (Entity ein : getInputs())
				if (ein.hasOutput()) in += ein.getOutput() * inputs.get(ein);

			// Aggregated catalysts
			double inCat = 0.0; // Neutral by default
			for (Entity ecat : catalyst)
				if (ecat.hasOutput()) inCat += ecat.getOutput();

			// Aggregated regulation
			double inRegPos = 0, inRegNeg = 0, inRegReq = 0; // Neutral by default
			int countPos = 0, countNeg = 0, countReq = 0;
			for (Entity ereg : regulator.keySet()) {
				if (ereg.hasOutput()) {
					double inReg = ereg.getOutput();
					RegulationType regType = regulator.get(ereg);

					switch (regType) {

					case PositiveRegulation:
						inRegPos += inReg;
						countPos++;
						break;

					case NegativeRegulation:
						inRegNeg += inReg;
						countNeg++;
						break;

					case Requirement:
						inRegReq += inReg;
						countReq++;
						break;
					}
				}
			}

			// Transfer function
			if (Double.isInfinite(in) || Double.isNaN(in)) output = Double.NaN; // Nothing in input? => Cannot calculate output
			else {
				double z = sigm(in);
				double cat = 2.0 * sigm(inCat);

				// Only active if there are inputs
				double regPos = 1.0, regNeg = 1.0, regReq = 1.0;
				if (countPos > 0) regPos = 1 + sigm(inRegPos);
				if (countNeg > 0) regNeg = 1 - sigm(inRegNeg);
				if (countReq > 0) regReq = sigm(inRegReq);

				output = 2.0 * (z * cat * regPos * regNeg * regReq) - 1.0;
			}
		}

		if (debug) System.out.println(output + "\tfixed:" + isFixed() + "\tid:" + id + "\ttype:" + getClass().getSimpleName() + "\tname:" + name);
		return output;
	}

	public HashSet getCatalyst() {
		return catalyst;
	}

	public Collection getInputs() {
		return inputs.keySet();
	}

	public HashSet getOutputs() {
		return outputs;
	}

	public HashMap getRegulator() {
		return regulator;
	}

	@Override
	public boolean isReaction() {
		return true;
	}

	/**
	 * Scale weights so that they add to 1
	 */
	public void scaleWeights() {
		// One input? Nothing to do
		if (inputs.size() <= 1) return;

		// Sum
		double sum = 0;
		for (Double in : inputs.values())
			sum += Math.abs(in);

		// Is it scaled? Nothing to do
		if (sum < MAX_WEIGHT_SUM) return;

		// New hash with scaled weights
		HashMap newInputs = new HashMap();
		for (Entity e : inputs.keySet()) {
			double weight = inputs.get(e) / sum;
			newInputs.put(e, weight);
		}

		// Replace hash
		inputs = newInputs;

	}

	double sigm(double x) {
		return 1.0 / (1.0 + Math.exp(-Entity.BETA * x));
	}

	@Override
	public String toString() {
		return toString(0, new HashSet());
	}

	@Override
	public String toString(int tabs, HashSet done) {
		done.add(this);

		StringBuilder sb = new StringBuilder();
		sb.append(Gpr.tabs(tabs) + getClass().getSimpleName() + "[" + id + "]: " + name + "\n");

		if (!inputs.isEmpty()) {
			sb.append(Gpr.tabs(tabs + 1) + "Inputs:\n");
			for (Entity e : getInputs()) {
				if (done.contains(e)) sb.append(Gpr.tabs(tabs + 2) + e.toStringSimple() + "\n");
				else {
					done.add(e);
					sb.append(e.toString(tabs + 2, done) + "\n");
				}
			}
		}

		if (!outputs.isEmpty()) {
			sb.append(Gpr.tabs(tabs + 1) + "Outputs:\n");
			for (Entity e : outputs) {
				if (done.contains(e)) sb.append(Gpr.tabs(tabs + 2) + e.toStringSimple() + "\n");
				else {
					done.add(e);
					sb.append(e.toString(tabs + 2, done) + "\n");
				}
			}
		}

		if (!catalyst.isEmpty()) {
			sb.append(Gpr.tabs(tabs + 1) + "Catalysts:\n");
			for (Entity e : catalyst) {
				if (done.contains(e)) sb.append(Gpr.tabs(tabs + 2) + e.toStringSimple() + "\n");
				else {
					done.add(e);
					sb.append(e.toString(tabs + 2, done) + "\n");
				}
			}
		}

		if (!regulator.isEmpty()) {
			sb.append(Gpr.tabs(tabs + 1) + "Regulator:\n");
			for (Entity e : regulator.keySet()) {
				if (done.contains(e)) sb.append(Gpr.tabs(tabs + 2) + e.toStringSimple() + "(" + regulator.get(e) + ")\t" + "\n");
				else {
					done.add(e);
					sb.append(e.toString(tabs + 2, done) + "(" + regulator.get(e) + ")\t" + "\n");
				}
			}
		}

		return sb.toString();
	}

	/**
	 * Transfer function
	 * @param h
	 * @return
	 */
	protected double transferFunction(double h, double inCat, double inRegPos, double inRegNeg, double inRegReq) {
		// Non-linear functions
		double z = 1.0 / (1.0 + Math.exp(-Entity.BETA * h));
		double cat = 2.0 / (1.0 + Math.exp(-Entity.BETA * inCat));
		double regPos = 1 + 1.0 / (1.0 + Math.exp(-Entity.BETA * inRegPos));
		double regNeg = 1 - 1.0 / (1.0 + Math.exp(-Entity.BETA * inRegNeg));
		double regReq = 1.0 / (1.0 + Math.exp(-Entity.BETA * inRegReq));

		return 2.0 * (z * cat * regPos * regNeg * regReq) - 1.0;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy