Newer
Older
import java.util.List;
import java.util.ArrayList;

Peter Wagenhuber
committed
import java.util.Iterator;

Peter Wagenhuber
committed
import java.lang.Math;

Peter Wagenhuber
committed
import java.util.ConcurrentModificationException;
private List<Channel> chanlist;
private List<Droplet> dropletList;
public Nloc (List<Channel> chanlist) {
this.chanlist = chanlist;
this.dropletList = new ArrayList<Droplet>();
}
public Pump getShortestSequence(String[] modulesToVisit) {

Peter Wagenhuber
committed
Pump pump = this.getPump();

Peter Wagenhuber
committed
List<List<SequenceTuple>> possibleSequences;
List<Droplet> dropletSequence;
List<SequenceTuple> currentSeq;

Peter Wagenhuber
committed

Peter Wagenhuber
committed
try {
possibleSequences = getPossibleSequences(modulesToVisit);

Peter Wagenhuber
committed
// sort the list of possible sequences according to the sequences length
possibleSequences.sort((a,b) -> a.size() - b.size());

Peter Wagenhuber
committed
do {
pump.removeAllDroplets();
currentSeq = possibleSequences.remove(0);
int min = 0, max = 0;
for (SequenceTuple stl: currentSeq) {
int tmp = stl.getMinPos();
if (tmp > max) max = tmp;
if (tmp < min) min = tmp;
}
int span = max - min;
pump.setSteps(span);
currentSeq = setTuplePumpoffsetToPumpPosition(currentSeq, min);
dropletSequence = generateDropletListRecursive(new ArrayList<Droplet>(), currentSeq, currentSeq.get(0).getMinPos());
currentSeq = resetPayloadDropletPosition(currentSeq);
} while (!possibleSequences.isEmpty() &&
dropletSequence.size() != currentSeq.size() &&

Peter Wagenhuber
committed
!this.sequenceFunctinal());

Peter Wagenhuber
committed
if (dropletSequence.size() != currentSeq.size()) {
pump = null;

Peter Wagenhuber
committed
}

Peter Wagenhuber
committed

Peter Wagenhuber
committed
} catch(NoSuchModuleException nsme) {
System.out.println(nsme.getMessage());

Peter Wagenhuber
committed
System.exit(1);

Peter Wagenhuber
committed
}
return pump;

Peter Wagenhuber
committed

Peter Wagenhuber
committed
public void setDropletList(List<Droplet> dropletList) {
this.dropletList = dropletList;
}

Peter Wagenhuber
committed
public List<SequenceTuple> resetPayloadDropletPosition(
List<SequenceTuple> currentSequence) {
for (SequenceTuple stup: currentSequence) {
if (stup.getDroplet().getType() == DropletType.PAYLOAD) {
stup.setMinPos(0);
stup.setMaxPos(0);
}
}
return currentSequence;
}
public List<SequenceTuple> setTuplePumpoffsetToPumpPosition(
List<SequenceTuple> currentSequence, int minOffset) {

Peter Wagenhuber
committed
for (SequenceTuple stup: currentSequence) {
int pumpOffsetMin = stup.getMinPos();
stup.setMinPos(pumpOffsetMin + Math.abs(minOffset));
int pumpOffsetMax = stup.getMaxPos();
stup.setMaxPos(pumpOffsetMax + Math.abs(minOffset));
}
return currentSequence;
}

Peter Wagenhuber
committed
public List<Droplet> generateDropletListRecursive(

Peter Wagenhuber
committed
List<Droplet> dropletList, List<SequenceTuple> currentSequence,
int currentPos) {
if (dropletList.size() == currentSequence.size()) {
return dropletList;
} else {
if (dropletList.isEmpty() ||
!Position.isOccupied(dropletList, currentPos)) {
SequenceTuple tmpTup = currentSequence.get(dropletList.size());
Droplet tmpDr = tmpTup.getDroplet();
tmpDr.setPosition(new Position(this.getPump(), currentPos));
Channel pumpOutlet = tmpTup.getPumpOutlet();
tmpDr.setPumpOutlet(pumpOutlet);

Peter Wagenhuber
committed
dropletList.add(tmpDr);
// set current pos to minimum of the following sequence tuple
if (dropletList.size() < currentSequence.size()) {
currentPos =
currentSequence.get(dropletList.size()).getMinPos();
}
return generateDropletListRecursive(dropletList, currentSequence,
currentPos);
} else {
SequenceTuple tmpTup = currentSequence.get(dropletList.size());
if (currentPos < tmpTup.getMaxPos()) {
currentPos++;
return generateDropletListRecursive(dropletList, currentSequence,
currentPos);
} else {
ListIterator<Droplet> revIter = dropletList.listIterator(
dropletList.size());
boolean found = false;
while (!found && revIter.hasPrevious()) {
Droplet tmpDr = revIter.previous();
SequenceTuple tmpTup1 = currentSequence.get(dropletList.size() - 1);
if (tmpDr.getPosition().getSteps() == currentPos &&
currentPos < tmpTup1.getMaxPos()) {
found = true;
}

Peter Wagenhuber
committed
revIter.remove();
}
if (found) {
currentPos++;
return generateDropletListRecursive(dropletList, currentSequence,
currentPos);
} else {
return dropletList;
}

Peter Wagenhuber
committed
}
}
}
}
public List<List<SequenceTuple>> getPossibleSequences(
String[] modulesToVisit) throws NoSuchModuleException {
Droplet payloadDroplet = new Droplet(DropletType.PAYLOAD,"p");
List<List<SequenceTuple>> sequences = new ArrayList<List<SequenceTuple>>();
List<List<Channel>> pathlist = this.getAllPaths();
try {
// Create initial list of possilbe sequences
List<Channel> moduleChanList =
this.getModulesByName(Arrays.asList(modulesToVisit));
List<Channel> payloadPath = this.getDesiredPath(moduleChanList, pathlist);
SequenceTuple plt = new SequenceTuple(payloadDroplet,payloadPath,0,0);
Channel previousBifurcation = plt.getPreviousBifurcation();
plt.setCurrentBifurcation(previousBifurcation);
List<SequenceTuple> s1 = new ArrayList<SequenceTuple>();
s1.add(plt);
sequences.add(s1);
List<List<SequenceTuple>> oldsequences;
do {
oldsequences = new ArrayList<List<SequenceTuple>>(sequences);
//System.out.println("oldseq: " + oldsequences);
//System.out.println("");
List<List<SequenceTuple>> tmpSeqs = new ArrayList<List<SequenceTuple>>();
//System.out.println("sequences before for computing: " + sequences);
//System.out.println("");
for (List<SequenceTuple> stl: sequences) {
tmpSeqs.addAll(
}
sequences = tmpSeqs;
//System.out.println("sequences after for computing: " + sequences);
//System.out.println("");
//System.out.println("");
} while (sequences.size() != oldsequences.size());
} catch (Exception e) {
System.out.println( e.getMessage());
}
return sequences;
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
}
//public List<List<SequenceTuple>> getPossibleSequences(
// String[] modulesToVisit) throws NoSuchModuleException {
// Droplet payloadDroplet = new Droplet(DropletType.PAYLOAD,"p");
// List<List<SequenceTuple>> sequences = new ArrayList<List<SequenceTuple>>();
// List<List<Channel>> pathlist = this.getAllPaths();
// try {
// // Create initial list of possilbe sequences
// List<Channel> moduleChanList =
// this.getModulesByName(Arrays.asList(modulesToVisit));
// List<Channel> payloadPath = this.getDesiredPath(moduleChanList, pathlist);
// SequenceTuple plt = new SequenceTuple(payloadDroplet,payloadPath,0,0);
// Channel previousBifurcation = plt.getPreviousBifurcation();
// plt.setCurrentBifurcation(previousBifurcation);
// List<SequenceTuple> s1 = new ArrayList<SequenceTuple>();
// s1.add(plt);
// sequences.add(s1);
// List<Channel> bifurcationList = this.getBifurcationList(payloadPath);
// // actually compile list of sequences going through the bifurcations from
// // "end" to "start"
// //for (int i = bifurcationList.size() - 1; i >= 0; --i) {
// //Channel currentBifurcation = bifurcationList.get(i);
// List<List<SequenceTuple>> oldsequences;
// do {
// oldsequences = new ArrayList<List<SequenceTuple>>(sequences);
// List<List<SequenceTuple>> tmpSeqs = new ArrayList<List<SequenceTuple>>();
// for (List<SequenceTuple> stl: sequences) {
// tmpSeqs.addAll(
// this.computeSequences(stl));
// }
// sequences = tmpSeqs;
// } while (sequences.size() != oldsequences.size());
// //}
// } catch (Exception e) {
// System.out.println( e.getMessage());
// }
// return sequences;
// }
public void addChannel(Channel chan) {
chanlist.add(chan);
}
public Sink getSink() {
Sink sink = null;
for (Channel chan : chanlist) {
if (chan instanceof Sink) sink = (Sink)chan;
}
return sink;
}
public Pump getPump() {
Pump pump = null;
for (Channel chan : chanlist) {
if (chan instanceof Pump) pump = (Pump)chan;
}
return pump;
}
public boolean sequenceFunctinal() {

Peter Wagenhuber
committed
this.dropletList = this.getPump().getDropletList();
boolean works = true;

Peter Wagenhuber
committed
while (!allDropletsInSink()) {
//for(int i = 0; i < 10; i++) {
try {
this.moveDroplets();
} catch (CoalescedDropletException e) {
works = false;
System.out.println("Coalesce! Droplet: " + e.getDroplet().getName() + " Position: " + e.getDroplet().getPosition().getChannel().getName());
break;
}
}
return works;
}
public List<Droplet> getDropletList() {
return dropletList;
}
public void moveDroplets() throws CoalescedDropletException {

Peter Wagenhuber
committed
Droplet[] drArr = this.dropletList.toArray(new Droplet[dropletList.size()]);
for (Droplet dr : drArr) {

Peter Wagenhuber
committed
//System.out.println("Droplet: " + dr.getName() + " Position: " + dr.getPosition().getChannel().getName() + ":" + dr.getPosition().getSteps());

Peter Wagenhuber
committed
//System.out.println("Droplet: " + dr.getName() + " Position: " + dr.getPosition().getChannel().getName() + ":" + dr.getPosition().getSteps());

Peter Wagenhuber
committed
//System.out.println("");

Peter Wagenhuber
committed
for (Droplet dr: drArr) {

Peter Wagenhuber
committed
//System.out.println("Droplet: " + dr.getName() + " Position: " + dr.getPosition().getChannel().getName() + ":" + dr.getPosition().getSteps());

Peter Wagenhuber
committed
if (dr.coalesce()) {
throw new CoalescedDropletException(dr);
}

Peter Wagenhuber
committed
public List<Channel> getBifurcationList(List<Channel> desiredPath) {

Peter Wagenhuber
committed
List<Channel> bfList = new ArrayList<Channel>();
for (Channel ch: desiredPath) {

Peter Wagenhuber
committed
bfList.add(ch);

Peter Wagenhuber
committed
}

Peter Wagenhuber
committed
}
return bfList;

Peter Wagenhuber
committed
}
public static int getPayloadPathlength(List<Channel> path) {
int len = 0;
for (Channel ch: path) {
len += ch.getPSteps();
}
return len;
}
public static int getHeaderPathlength(List<Channel> path) {
int len = 0;
for (Channel ch: path) {
len += ch.getHSteps();
}
return len;
}

Peter Wagenhuber
committed
public boolean allDropletsInSink() {
boolean allInSink = true;
for (Droplet dr : dropletList) {
allInSink &= dr.isInSink();
}
return allInSink;
public List<Channel> getDesiredPath(List<Channel> modules,
List<List<Channel>> pathlist) throws NlocStructureException {
List<Channel> found = new ArrayList<Channel>();
for (List<Channel> path : pathlist) {
List<Channel> modlist = new ArrayList<Channel>();
for (Channel ch: path) {
if (ch instanceof Module) {
modlist.add(ch);
}
}
if (modules.equals(modlist) && found.isEmpty()) {
found = path;
} else if (modules.equals(modlist) && !found.isEmpty()) {
throw new NlocStructureException("Paths are not unique!");
}
if (found.isEmpty()) {
throw new NlocStructureException("No path found that covers all given Modules");
return found;
}
public Channel getModuleByName(String name) throws NoSuchModuleException {
Channel found = null;
for (Channel ch: chanlist) {
if (ch.getName().equals(name)) {
found = ch;
}
}
if (found == null) throw new NoSuchModuleException(name);
return found;
}
public List<Channel> getModulesByName(List<String> names)
throws NoSuchModuleException {
List<Channel> ret = new ArrayList<Channel>();
for (String name : names) {
Channel ch = getModuleByName(name);
if (name != null) {
ret.add(ch);
public List<List<Channel>> getAllPathsFromTo(Channel from, Channel to) {
List<List<Channel>> pl = new ArrayList<List<Channel>>();
List<Channel> path = new ArrayList<Channel>();
getAllPathsRecursive(from, to, path, pl);
return pl;
}
public List<List<Channel>> getAllPaths() {
List<List<Channel>> pl = new ArrayList<List<Channel>>();
List<Channel> path = new ArrayList<Channel>();
getAllPathsRecursive(this.getPump(), this.getSink(), path, pl);
private void getAllPathsRecursive(Channel chan, Channel end, List<Channel> path, List<List<Channel>> pathlist) {
if (chan.equals(end)) {
pathlist.add(path);
} else {
for (Channel ch : chan.getChildrenList()) {
getAllPathsRecursive(ch, end, new ArrayList<Channel>(path), pathlist);
public List<List<SequenceTuple>> computeSequences(List<SequenceTuple> seqTup){

Peter Wagenhuber
committed
List<List<SequenceTuple>> seqTupList = new ArrayList<List<SequenceTuple>>();
//SequenceTuple currentSeqTup = seqTup.get(0);
getSequencesAtBifurcationRecursive(seqTup, 0, seqTupList);
//System.out.println("returning " + seqTupList);

Peter Wagenhuber
committed
return seqTupList;
}
private void getSequencesAtBifurcationRecursive(
List<SequenceTuple> seqTupList, int currentSeqTupIndex,
List<List<SequenceTuple>> possibleSequences) {
SequenceTuple currentSeqTup = seqTupList.get(currentSeqTupIndex);
Channel currentBifurcation = currentSeqTup.getCurrentBifurcation();
//currentSeqTup.setCurrentBifurcation(currentSeqTup.getPreviousBifurcation());

Peter Wagenhuber
committed
List<Channel> dropletPath = currentSeqTup.getPath();
Channel bifurcSuccessor =
dropletPath.get(dropletPath.indexOf(currentBifurcation) + 1);
// bifurcation priority: prio = 0 if default channel; prio >= 1 if not
// default and threrfore header droplet needed
int prio = currentBifurcation.getChildrenList().indexOf(bifurcSuccessor);

Peter Wagenhuber
committed
if (currentSeqTup.equals(seqTupList.get(seqTupList.size() - 1))) {
// if at last sequence tuple add another header droplet tuple if needed
// and add the list of sequence tuples to the possible sequences list
if (dropletPath.contains(currentBifurcation) && prio > 0 &&
!(currentBifurcation instanceof Pump)) {
//System.out.println("we need header");

Peter Wagenhuber
committed
// we need header droplet

Peter Wagenhuber
committed
// cirst check all possible paths of header droplets

Peter Wagenhuber
committed
Channel defaultChan = currentBifurcation.getChildrenList().get(0);

Peter Wagenhuber
committed
List<List<Channel>> pathList =

Peter Wagenhuber
committed
this.getAllPathsFromTo(dropletPath.get(0), defaultChan);
currentSeqTup.
setCurrentBifurcation(currentSeqTup.getPreviousBifurcation());

Peter Wagenhuber
committed

Peter Wagenhuber
committed
for (List<Channel> path: pathList) {
//System.out.println("iterating through pathlist");
List<SequenceTuple> tmp = new ArrayList<SequenceTuple>();
for (SequenceTuple stup: seqTupList) {
tmp.add(new SequenceTuple(stup));
}

Peter Wagenhuber
committed
SequenceTuple tmpTuple =
new SequenceTuple(new Droplet(DropletType.HEADER,"h"),path,
currentBifurcation);
tmpTuple.setCurrentBifurcation(tmpTuple.getPreviousBifurcation());

Peter Wagenhuber
committed
tmp.add(tmpTuple);
//System.out.println("tmp:");
//for (SequenceTuple tstu: tmp) {
// System.out.println("tstu: " + tstu);
//}

Peter Wagenhuber
committed

Peter Wagenhuber
committed
// calculate and set pump offsets

Peter Wagenhuber
committed
int minPos = currentSeqTup.getMinPos();
int maxPos = currentSeqTup.getMaxPos();
List<Channel> pathToCurrentBifurcation =
dropletPath.subList(0,dropletPath.indexOf(currentBifurcation) + 1);

Peter Wagenhuber
committed
int pathLenCurrDroplet = 0;
if (currentSeqTup.getDroplet().getType() == DropletType.HEADER) {
pathLenCurrDroplet = getHeaderPathlength(pathToCurrentBifurcation);
} else {
pathLenCurrDroplet = getPayloadPathlength(pathToCurrentBifurcation);
}
int maxPathLenNewDroplet = getHeaderPathlength(path);
int minPathLenNewDroplet =
maxPathLenNewDroplet - defaultChan.getHSteps() + 1;
//System.out.println("minPos: " + minPos + " pathLenCurrDroplet: " +
// pathLenCurrDroplet + " minPathLenNewDroplet: " + minPathLenNewDroplet);

Peter Wagenhuber
committed
int newTupleMinPos =
minPos - (pathLenCurrDroplet - minPathLenNewDroplet);
int newTupleMaxPos =
maxPos - (pathLenCurrDroplet - maxPathLenNewDroplet);

Peter Wagenhuber
committed
tmpTuple.setMinPos(newTupleMinPos);
tmpTuple.setMaxPos(newTupleMaxPos);

Peter Wagenhuber
committed
possibleSequences.add(tmp);
//System.out.println("adding tmp to possible sequences " + possibleSequences);

Peter Wagenhuber
committed
}

Peter Wagenhuber
committed
} else {
//System.out.println("else");

Peter Wagenhuber
committed
possibleSequences.add(seqTupList);

Peter Wagenhuber
committed
}
//possibleSequences.add(seqTupList);
if (dropletPath.contains(currentBifurcation) && prio > 0 &&
!(currentBifurcation instanceof Pump)) {
//System.out.println("we need header for: " + currentSeqTup.getDroplet() + " " + currentSeqTup.getDroplet().getName());
//System.out.println("At: " + currentBifurcation.getName());
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
// we need header droplet
// cirst check all possible paths of header droplets
Channel defaultChan = currentBifurcation.getChildrenList().get(0);
List<List<Channel>> pathList =
this.getAllPathsFromTo(dropletPath.get(0), defaultChan);
for (List<Channel> path: pathList) {
//List<SequenceTuple> tmp = new ArrayList<SequenceTuple>(seqTupList);
List<SequenceTuple> tmp = new ArrayList<SequenceTuple>();
for (SequenceTuple stup: seqTupList) {
tmp.add(new SequenceTuple(stup));
}
SequenceTuple tmpTuple =
new SequenceTuple(new Droplet(DropletType.HEADER,"h"),path,
currentBifurcation);
tmpTuple.setCurrentBifurcation(tmpTuple.getPreviousBifurcation());
tmp.add(seqTupList.indexOf(currentSeqTup) + 1, tmpTuple);
// calculate and set pump offsets
int minPos = currentSeqTup.getMinPos();
int maxPos = currentSeqTup.getMaxPos();
List<Channel> pathToCurrentBifurcation =
dropletPath.subList(0,dropletPath.indexOf(currentBifurcation) + 1);
int pathLenCurrDroplet = 0;
if (currentSeqTup.getDroplet().getType() == DropletType.HEADER) {
pathLenCurrDroplet = getHeaderPathlength(pathToCurrentBifurcation);
} else {
pathLenCurrDroplet = getPayloadPathlength(pathToCurrentBifurcation);
}
int maxPathLenNewDroplet = getHeaderPathlength(path);
int minPathLenNewDroplet =
maxPathLenNewDroplet - defaultChan.getHSteps() + 1;
//System.out.println("minPos: " + minPos + " pathLenCurrDroplet: " +
// pathLenCurrDroplet + " minPathLenNewDroplet: " + minPathLenNewDroplet);
int newTupleMinPos =
minPos - (pathLenCurrDroplet - minPathLenNewDroplet);
int newTupleMaxPos =
maxPos - (pathLenCurrDroplet - maxPathLenNewDroplet);
tmpTuple.setMinPos(newTupleMinPos);
tmpTuple.setMaxPos(newTupleMaxPos);
getSequencesAtBifurcationRecursive(tmp, currentSeqTupIndex + 1,
possibleSequences);
}
} else {
getSequencesAtBifurcationRecursive(seqTupList, currentSeqTupIndex + 1,
possibleSequences);
}