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

edu.cmu.tetrad.search.work_in_progress.DMSearch Maven / Gradle / Ivy

There is a newer version: 7.6.5
Show newest version
package edu.cmu.tetrad.search.work_in_progress;

import edu.cmu.tetrad.data.CovarianceMatrix;
import edu.cmu.tetrad.data.DataSet;
import edu.cmu.tetrad.data.Knowledge;
import edu.cmu.tetrad.graph.*;
import edu.cmu.tetrad.search.Fges;
import edu.cmu.tetrad.search.IndependenceTest;
import edu.cmu.tetrad.search.score.SemBicScore;
import edu.cmu.tetrad.search.test.IndTestFisherZ;
import org.apache.commons.math3.linear.SingularMatrixException;

import java.io.File;
import java.io.FileOutputStream;
import java.io.PrintStream;
import java.util.*;

/**
 * Implements the DM search.
 *
 * @author Alexander Murray-Watters
 */
public class DMSearch {
    private int[] inputs;
    private int[] outputs;

    //alpha value for sober's criterion.
    private double alphaSober = .05;

    //Alpha value for pc.
    private double alphaPC = .05;


    //Starting ges penalty discount.
    private double gesDiscount = 10;
    private int gesDepth = 0;

    //Minimum ges penalty discount to use in recursive search.
    private int minDiscount = 4;

    //If true, use GES, else use PC.
    private boolean useGES = true;

    //Lets the user select a subset of the inputs in the dataset to search over.
    //If not subseting, should be set to the entire input set.
    private int[] trueInputs;
    private DataSet data;
    private CovarianceMatrix cov;
    private LatentStructure dmStructure;

    public void setMinDiscount(int minDiscount) {
        this.minDiscount = minDiscount;
    }

    public int getMinDepth() {
        return (this.minDiscount);
    }

    public int getGesDepth() {
        return (gesDepth);
    }

    public void setGesDepth(int gesDepth) {
        this.gesDepth = gesDepth;
    }

    public int[] getTrueInputs() {
        return (this.trueInputs);
    }

    public void setTrueInputs(int[] trueInputs) {
        this.trueInputs = trueInputs;
    }

    public DataSet getData() {
        return (this.data);
    }

    public void setData(DataSet data) {
        this.data = data;
    }

    public int[] getInputs() {
        return (inputs);
    }

    public void setInputs(int[] inputs) {
        this.inputs = inputs;
    }

    public int[] getOutputs() {
        return (outputs);
    }

    public void setOutputs(int[] outputs) {
        this.outputs = outputs;
    }

    public LatentStructure getDmStructure() {
        return (dmStructure);
    }

    public void setDmStructure(LatentStructure structure) {
        this.dmStructure = structure;
    }

    public void setAlphaSober(double alpha) {
        this.alphaSober = alpha;
    }

    public void setAlphaPC(double alpha) {
        this.alphaPC = alpha;
    }

    public void setDiscount(double discount) {
        this.gesDiscount = discount;
    }

    public void setUseFgES(boolean set) {
        this.useGES = set;
    }


    public Graph search() {

        int[] trueInputs = getTrueInputs();

        DataSet data = getData();

        //TODO: Break stuff below here into seperate fuct/classes.
        this.cov = new CovarianceMatrix(data);


        Knowledge knowledge = new Knowledge(data.getVariableNames());

        //Forbids edges from outputs to inputs.
        for (int i : getInputs()) {
            knowledge.addToTier(0, data.getVariable(i).getName());
        }

        for (int i : getOutputs()) {
            knowledge.addToTier(1, data.getVariable(i).getName());
        }

        knowledge.setTierForbiddenWithin(0, true);
        knowledge.setTierForbiddenWithin(1, true);

        Set inputString = new HashSet();

        HashSet actualInputs = new HashSet();
        for (int i = 0; i < trueInputs.length; i++) {
            actualInputs.add(trueInputs[i]);
        }


        for (int i : inputs) {
            if (actualInputs.contains(i)) {

                inputString.add(data.getVariable(i).getName());
            }
        }

        Graph pattern = new EdgeListGraph();

        if (useGES == true) {
            Fges ges = new Fges(new SemBicScore(cov));

            pattern = recursiveGES(pattern, knowledge, this.gesDiscount, getMinDepth(), data, inputString);
        } else {
            this.cov = new CovarianceMatrix(data);
//            Pc pc = new Pc(new IndTestFisherZ(cov, this.alphaPC));
//            pc.setKnowledge(knowledge);
//            pc.setDepth(0);
            System.out.println("Running PC Search");
//            pattern = pc.search();
            double penalty = 2;


//           TODO: Alternative to using built in PC. Needs a fix so that all nodes added to pattern are looked at in applyDmSearch
//            ExecutorService executorService = Executors.newFixedThreadPool(4); // number of threads

            IndTestFisherZ ind = new IndTestFisherZ(cov, this.alphaPC);
            for (int i = 0; i < getInputs().length; i++) {
                if (!pattern.containsNode(data.getVariable(i))) {
                    pattern.addNode(data.getVariable(i));
                }

                if (actualInputs.contains(i)) {
                    for (int j = getInputs().length; j < data.getNumColumns(); j++) {
                        if (!pattern.containsNode(data.getVariable(j))) {
                            pattern.addNode(data.getVariable(j));
                        }

//                    System.out.println(i);
//                    System.out.println(j);
                        if (ind.checkIndependence(data.getVariable(i), data.getVariable(j)).isDependent()) {
                            pattern.addDirectedEdge(data.getVariable(i), data.getVariable(j));
                        }
                    }
                }
            }

            System.out.println("Running DM search");
            applyDmSearch(pattern, inputString, penalty);
        }

        return (getDmStructure().latentStructToEdgeListGraph(getDmStructure()));

    }

    public LatentStructure applyDmSearch(Graph pattern, Set inputString, double penalty) {

        List> outputParentsList = new ArrayList>();
        final List patternNodes = pattern.getNodes();

//        TODO: add testcase to see how sort compares 10, 11, 1, etc.
        java.util.Collections.sort(patternNodes, new Comparator() {
            public int compare(Node node1, Node node2) {
//TODO: string length error here. Fix.

                if (node1.getName().length() > node2.getName().length()) {
                    return (1);
                } else if (node1.getName().length() < node2.getName().length()) {
                    return (-1);
                } else {
                    int n1 = Integer.parseInt(node1.getName().substring(1));
                    int n2 = Integer.parseInt(node2.getName().substring(1));
                    return (n1 - n2);
                }
            }
        });

        System.out.println("Sorted patternNodes");
        //constructing treeSet of output nodes.
        SortedSet outputNodes = new TreeSet();
        for (int i : getOutputs()) {

//            System.out.println("patternNodes");
//            System.out.println(patternNodes);

//            System.out.println("i");
//            System.out.println(i);
            outputNodes.add(patternNodes.get(i));
        }

        System.out.println("Got output nodes");

//        System.out.println(outputNodes);

        //Constructing list of output node parents.
        for (Node node : outputNodes) {
            outputParentsList.add(new TreeSet(getInputParents(node, inputString, pattern)));
        }

        System.out.println("Created list of output node parents");
        int sublistStart = 1;
        int nLatents = 0;

        LatentStructure structure = new LatentStructure();

        //Creating set of nodes with same input sets.
        // And adding both inputs and outputs to their respective latents.
        for (Set set1 : outputParentsList) {

            TreeSet sameSetParents = new TreeSet(new Comparator() {
                public int compare(Node node1, Node node2) {

                    if (node1.getName().length() > node2.getName().length()) {
                        return (1);
                    } else if (node1.getName().length() < node2.getName().length()) {
                        return (-1);
                    } else {
                        int n1 = Integer.parseInt(node1.getName().substring(1));
                        int n2 = Integer.parseInt(node2.getName().substring(1));
                        return (n1 - n2);
                    }
                }
            });

            List> nextSet = outputParentsList.subList(sublistStart, outputParentsList.size());
            //If no next set, then just add var.
            if (nextSet.isEmpty()) {
//                for (int head = 0; head < (set1.size()); head++) {
                sameSetParents.addAll(set1);
//                }
            }
            for (Set set2 : nextSet) {
                if (!(set1.size() == 0 || set2.size() == 0) && set1.equals(set2)) {
//                    for (int head = 0; head < (set1.size()); head++) {
                    sameSetParents.addAll(set1);
//                    }
                } else if (set1.size() > 0) {
//                    for (int head = 0; head < (set1.size()); head++) {
                    sameSetParents.addAll(set1);
//                    }
                }
            }

            if (sameSetParents.size() > 0) {
                //Creates a new latent with a size 1 less than actually present.
                GraphNode tempLatent = new GraphNode("L" + nLatents);

                if (!setContained(structure, structure.inputs.keySet(), sameSetParents) || structure.inputs.isEmpty()) {
                    structure.latents.add(tempLatent);
                    structure.inputs.put(tempLatent, sameSetParents);
                    nLatents++;
                } else {
                    continue;
                }

                // TODO: Spin off into own function, which adds the output nodes
                //Adding Outputs to their Map.
                for (Node node : outputNodes) {

                    if (new TreeSet(getInputParents(node, inputString, pattern)).equals(sameSetParents)) {

                        //If haven't created latent, then do so.
                        if (structure.outputs.get(tempLatent) == null) {
                            TreeSet outputNode = new TreeSet();
                            outputNode.add(node);
                            structure.outputs.put(tempLatent, outputNode);
                        }
                        //Otherwise, just add new node to set of output
                        // nodes for the given latent.
                        else {
                            structure.outputs.get(tempLatent).add(node);
                        }
                    }
                }
            }
            System.out.println("Completed starting point: " + sublistStart + " out of #" + outputParentsList.size() + " sets, and is " + set1.size() + " units large.");
            sublistStart++;
        }
        System.out.println("created initial sets");

        //Need to order latents by entryset value size (smallest to largest)
        //as Map only allows sorting by keyset size.
        TreeMap, Node> latentsSortedByInputSetSize = sortMapByValue(structure.inputs, structure.latents, structure);


        System.out.println("Finding initial latent-latent effects");


//        System.out.println(latentsSortedByInputSetSize);


        TreeSet inputs1 = new TreeSet(new Comparator() {
            public int compare(Node node1, Node node2) {

                if (node1.getName().length() > node2.getName().length()) {
                    return (1);
                } else if (node1.getName().length() < node2.getName().length()) {
                    return (-1);
                } else {
                    int n1 = Integer.parseInt(node1.getName().substring(1));
                    int n2 = Integer.parseInt(node2.getName().substring(1));
                    return (n1 - n2);
                }
            }
        });

        TreeSet inputs2 = new TreeSet(new Comparator() {
            public int compare(Node node1, Node node2) {

                if (node1.getName().length() > node2.getName().length()) {
                    return (1);
                } else if (node1.getName().length() < node2.getName().length()) {
                    return (-1);
                } else {
                    int n1 = Integer.parseInt(node1.getName().substring(1));
                    int n2 = Integer.parseInt(node2.getName().substring(1));
                    return (n1 - n2);
                }
            }
        });

        HashSet alreadyLookedAt = new HashSet();

        //Finding initial latent-latent Effects.
        for (int i = 0; i <= latentsSortedByInputSetSize.keySet().size(); i++) {

//          TODO: Need to only perform this test if haven't already looked at latent. (for latent 1).


            TreeSet> sortedInputs = new TreeSet>(new Comparator>() {
                public int compare(TreeSet o1, TreeSet o2) {
                    int size = o1.size() - o2.size();
                    if (size == 0) {
                        if (o1.equals(o2)) {
                            return (0);
                        } else {
                            return (o1.hashCode() - o2.hashCode());
                        }
                    } else {
                        return (size);
                    }
                }
            });

            sortedInputs.addAll(latentsSortedByInputSetSize.keySet());

            inputs1 = findFirstUnseenElement(sortedInputs, alreadyLookedAt, latentsSortedByInputSetSize);

            HashSet alreadyLookedAtInnerLoop = new HashSet();

            Node latent1 = latentsSortedByInputSetSize.get(inputs1);

            if (inputs1.first().getName().equals("alreadySeenEverything")) {
                continue;
            }

            for (int j = 0; j <= latentsSortedByInputSetSize.keySet().size(); j++) {


                TreeSet temp2 = new TreeSet>(new Comparator>() {
                    public int compare(TreeSet o1, TreeSet o2) {
                        int size = o1.size() - o2.size();
                        if (size == 0) {
                            if (o1.equals(o2)) {
                                return (0);
                            } else {
                                return (o1.hashCode() - o2.hashCode());
                            }
                        } else {
                            return (size);
                        }
                    }
                });

                inputs2 = findFirstUnseenElement(sortedInputs, alreadyLookedAtInnerLoop, latentsSortedByInputSetSize);


                Node latent2 = latentsSortedByInputSetSize.get(inputs2);

                if (inputs2.first().getName().equals("alreadySeenEverything")) {
                    continue;
                }

                alreadyLookedAtInnerLoop.add(latent2);


                if (latent1.equals(latent2) || structure.getInputs(latent2).equals(structure.getInputs(latent1))) {
                    continue;
                }


                //if latent1 is a subset of latent2...
                if (structure.getInputs(latent2).containsAll(structure.getInputs(latent1))) {
                    if (structure.latentEffects.get(latent1) == null) {
                        TreeSet latentEffects = new TreeSet(new Comparator() {
                            public int compare(Node node1, Node node2) {

                                if (node1.getName().length() > node2.getName().length()) {
                                    return (1);
                                } else if (node1.getName().length() < node2.getName().length()) {
                                    return (-1);
                                } else {
                                    int n1 = Integer.parseInt(node1.getName().substring(1));
                                    int n2 = Integer.parseInt(node2.getName().substring(1));
                                    return (n1 - n2);
                                }
                            }
                        });
                        latentEffects.add(latent2);
                        structure.latentEffects.put(latent1, latentEffects);
                    } else {
                        structure.latentEffects.get(latent1).add(latent2);
                    }
                    //Removes set of inputs from every other latent's input set.


                    latentsSortedByInputSetSize = removeSetInputs(structure, structure.getInputs(latent1),
                            structure.getInputs(latent2).size(), latent2, latentsSortedByInputSetSize);

                }
            }
            alreadyLookedAt.add(latent1);

        }


//        Ensuring no nulls in latenteffects map.
        SortedSet emptyTreeSet = new TreeSet(new Comparator() {
            public int compare(Node node1, Node node2) {

                if (node1.getName().length() > node2.getName().length()) {
                    return (1);
                } else if (node1.getName().length() < node2.getName().length()) {
                    return (-1);
                } else {
                    int n1 = Integer.parseInt(node1.getName().substring(1));
                    int n2 = Integer.parseInt(node2.getName().substring(1));
                    return (n1 - n2);
                }
            }
        });

        for (Node latent : structure.getLatents()) {
            if (structure.latentEffects.get(latent) == null) {
                structure.latentEffects.put(latent, emptyTreeSet);
            }
        }

        System.out.println("Structure prior to Sober's step:");
//        System.out.println(structure);

        System.out.println("Applying Sober's step ");

        //Sober's step.
        for (Node latent : structure.getLatents()) {
            if (structure.latentEffects.containsKey(latent)) {
                for (Node latentEffect : structure.getLatentEffects(latent)) {
                    applySobersStep(structure.getInputs(latent), structure.getInputs(latentEffect),
                            structure.getOutputs(latent), structure.getOutputs(latentEffect),
                            pattern, structure, latent, latentEffect);
                }
            }
        }

        setDmStructure(structure);


        //Saves DM output in case is needed.
        File file = new File("src/edu/cmu/tetradproj/amurrayw/DM_output_" + "GES_penalty" + penalty + "_.txt");
        try {
            FileOutputStream out = new FileOutputStream(file);
            PrintStream outStream = new PrintStream(out);
            outStream.println(structure.latentStructToEdgeListGraph(structure));
        } catch (java.io.FileNotFoundException e) {
            System.out.println("Can't write to file.");

        }


        return (structure);

    }

    private TreeSet findFirstUnseenElement(TreeSet> set, HashSet alreadySeen, TreeMap map) {
        for (TreeSet currentSet : set) {
            if (!(alreadySeen.contains(map.get(currentSet))) && map.get(currentSet) != null) {
                return (currentSet);
            }
        }
        GraphNode end = new GraphNode("alreadySeenEverything");

        TreeSet seenEverything = new TreeSet();
        seenEverything.add(end);


        return (seenEverything);

    }


    private TreeSet nthElementOn(TreeSet set, int startingElementPos) {

        for (int i = 0; i < set.size() - startingElementPos; i++) {
            set = rest(set);
        }

        return (set);
    }

    //    Pulls head off of set and returns rest. (think cdr in lisp)
    private TreeSet> rest(TreeSet set) {
        set.remove(set.first());
        return (set);
    }

    //returns second set of nodes.(think cadr in lisp).
    private TreeSet> second(TreeSet> set) {

        TreeSet> secondNodeSet = new TreeSet<>();

        secondNodeSet = rest(set);

        secondNodeSet.first();

        return (secondNodeSet);

    }

    private boolean allEqual(SortedSet set1, SortedSet set2) {
        for (Node i : set1) {
            for (Node j : set2) {
                if (i.equals(j)) {
                    continue;
                } else {
                    return (false);
                }
            }
        }
        for (Node i : set2) {
            for (Node j : set1) {
                if (i.equals(j)) {
                    continue;
                } else {
                    return (false);
                }
            }
        }
        return (true);
    }

    // Uses previous runs of GES as new knowledge for a additional runs of GES with lower penalty discounts.
    private Graph recursiveGES(Graph previousGES, Knowledge knowledge, double penalty, double minPenalty, DataSet data, Set inputString) {

        for (Edge edge : previousGES.getEdges()) {
            knowledge.setRequired(edge.getNode1().getName(), edge.getNode2().getName());
        }

        previousGES = null;

        this.cov = new CovarianceMatrix(data);


        Fges ges = new Fges(new SemBicScore(cov));
        ges.setKnowledge(knowledge);

        Graph pattern = ges.search();


        //Saves GES output in case is needed.
        File file = new File("src/edu/cmu/tetradproj/amurrayw/ges_output_" + penalty + "_.txt");
        try {
            FileOutputStream out = new FileOutputStream(file);
            PrintStream outStream = new PrintStream(out);
            outStream.println(pattern);
        } catch (java.io.FileNotFoundException e) {
            System.out.println("Can't write to file.");

        }

        if (penalty > minPenalty) {
            applyDmSearch(pattern, inputString, penalty);
            return (recursiveGES(pattern, knowledge, penalty - 1, minPenalty, data, inputString));
        } else {
            applyDmSearch(pattern, inputString, penalty);
            return (pattern);
        }

    }


    //Finds any input set that mseps outputs for a pair of directly related latents, then adds input set to approp. set
    //Finally removes latent effect from list of latent effects.
    private void applySobersStep(SortedSet inputsLatent, SortedSet inputsLatentEffect,
                                 SortedSet outputsLatent, SortedSet outputsLatentEffect,
                                 Graph pattern, LatentStructure structure, Node latent, Node latentEffect) {

        Set latentList = new HashSet<>();

        latentList.addAll(inputsLatent);

        IndependenceTest test = new IndTestFisherZ(this.cov, this.alphaSober);

        boolean testResult = false;

        try {
            testResult = test.checkIndependence(outputsLatent.first(), outputsLatentEffect.first(), latentList).isIndependent();
        } catch (SingularMatrixException error) {
            error.printStackTrace();
            System.out.println("SingularMatrixException Error!!!!!! Evaluated as:");
            System.out.println(testResult);
            System.out.println("outputsLatent.first()");
            System.out.println(outputsLatent.first());
            System.out.println("outputsLatentEffect.first()");
            System.out.println(outputsLatentEffect.first());
        }
        if (testResult == true) {
            structure.latentEffects.get(latent).remove(latentEffect);
            structure.inputs.get(latentEffect).addAll(inputsLatent);
        }
    }

    // Removes subset of inputs from any latent's input set.
    //Inputs are latent structure, the set of inputs which are to be removed, the number of
    // inputs in the superset, and the identity of the latent they are a subset of,
    private TreeMap removeSetInputs(LatentStructure structure, SortedSet set, int sizeOfSuperset, Node latentForSuperset, TreeMap, Node> map) {
        for (Node latent : structure.latents) {
            if (structure.inputs.get(latent).equals(set)) {
                continue;
            } else {


                //Want to remove input set only if the target set is greater than the discovered superset or is superset.
                if (structure.inputs.get(latent).size() > sizeOfSuperset || latent.equals(latentForSuperset)) {
                    if (structure.inputs.get(latent).containsAll(set)) {

                        structure.inputs.get(latent).removeAll(set);

                    }

                }
            }
        }
        return (map);
    }

    //Returns true if a latent already contains the given input set.
    private boolean setContained(LatentStructure structure, Set latentSet, Set inputSet) {
        for (Node latent : latentSet) {
            if (structure.getInputs(latent).equals(inputSet)) {
                return (true);
            }
        }
        return (false);
    }

    @Override
    public boolean equals(Object obj) {
        return super.equals(obj);
    }

    private SortedSet copy(SortedSet orig) {
        SortedSet newset = new TreeSet();

        for (Object o : orig) {
            newset.add(o);

        }
        return (newset);
    }


    private TreeMap, Node> sortMapByValue(Map> inputMap, List latents, LatentStructure structure) {

        TreeMap, Node> sortedInputSets = new TreeMap, Node>(new Comparator>() {
            public int compare(SortedSet o1, SortedSet o2) {
                int size = o1.size() - o2.size();
                if (size == 0) {
                    if (o1.equals(o2)) {
                        return (0);
                    } else {
                        return (o1.hashCode() - o2.hashCode());
                    }
                } else {
                    return (size);
                }
            }
        });

        for (Node latent : latents) {
            TreeSet tempSet = new TreeSet<>(new Comparator() {
                public int compare(Node node1, Node node2) {

                    if (node1.getName().length() > node2.getName().length()) {
                        return (1);
                    } else if (node1.getName().length() < node2.getName().length()) {
                        return (-1);
                    } else {
                        int n1 = Integer.parseInt(node1.getName().substring(1));
                        int n2 = Integer.parseInt(node2.getName().substring(1));
                        return (n1 - n2);
                    }
                }
            });

            tempSet.addAll(inputMap.get(latent));

            sortedInputSets.put(tempSet, latent);
        }
        return (sortedInputSets);
    }


    //Making sure found nodes are actually inputs before adding as knowledge is now disabled.
    private Set getInputParents(Node node, Set inputString, Graph pattern) {
        Set actualInputs = new HashSet();
        for (Node posInput : pattern.getAdjacentNodes(node)) {
            if (inputString.contains(posInput.getName())) {
                actualInputs.add(posInput);
            }
        }
        return (actualInputs);
    }

    public class LatentStructure {
        List latents = new ArrayList();
        Map> inputs = new TreeMap>();
        Map> outputs = new TreeMap>();
        Map> latentEffects = new TreeMap>();


        public LatentStructure() {
        }

        public void addRecord(Node latent, SortedSet inputs, SortedSet outputs, SortedSet latentEffects) {
            if (latents.contains(latent)) throw new IllegalArgumentException();

            this.latents.add(latent);
            this.inputs.put(latent, inputs);
            this.outputs.put(latent, outputs);
            this.latentEffects.put(latent, latentEffects);
        }

        public void removeLatent(Node latent) {
            this.latents.remove(latent);
            this.inputs.remove(latent);
            this.outputs.remove(latent);
            this.latentEffects.remove(latent);
        }

        public List getLatents() {
            return new ArrayList(latents);
        }

        public boolean containsLatent(Node latent) {
            return latents.contains(latent);
        }

        public SortedSet getInputs(Node latent) {
            return new TreeSet(inputs.get(latent));
        }

        public SortedSet getOutputs(Node latent) {
            return new TreeSet(outputs.get(latent));
        }

        public SortedSet getLatentEffects(Node latent) {
            return new TreeSet(latentEffects.get(latent));
        }

        public String toString() {
            StringBuilder b = new StringBuilder();

            for (Node node : latents) {
                b.append("Latent:" + node + "\n "
                        + "Inputs:" + inputs.get(node) + "\n "
                        + "Outputs:" + outputs.get(node) + "\n "
                        + "Latent Effects:" + latentEffects.get(node) + "\t\n");
            }

            b.append("\n");
            return b.toString();
        }

        public Graph latentStructToEdgeListGraph(LatentStructure structure) {

            Graph structureGraph = new EdgeListGraph();


            for (Node latent : latents) {
                //Adding every node to graph.
                structureGraph.addNode(latent);

                for (Node input : inputs.get(latent)) {
                    structureGraph.addNode(input);
                }
                for (Node output : outputs.get(latent)) {
                    structureGraph.addNode(output);
                }

                //adding edges from inputs to latent.
                for (Node input : inputs.get(latent)) {
                    structureGraph.addDirectedEdge(input, latent);
                }

                //adding edges from latent to outputs.
                for (Node output : outputs.get(latent)) {
                    structureGraph.addDirectedEdge(latent, output);
                }

                //adding edges from latents to latents.
                if (latentEffects.get(latent) == null) {
                    continue;
                } else {
                    for (Node latentEff : latentEffects.get(latent)) {

                        if (!structureGraph.containsNode(latentEff)) {
                            structureGraph.addNode(latentEff);
                        }
                        structureGraph.addDirectedEdge(latent, latentEff);
                    }
                }
            }
            return (structureGraph);
        }


    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy