23import java.util.ArrayList;
24import java.util.Arrays;
25import java.util.HashMap;
26import java.util.HashSet;
27import java.util.LinkedHashMap;
30import java.util.Map.Entry;
32import java.util.TreeMap;
33import java.util.logging.Level;
34import java.util.stream.Collectors;
35import java.util.stream.IntStream;
37import org.apache.http.TruncatedChunkException;
38import org.openscience.cdk.graph.ShortestPaths;
39import org.openscience.cdk.interfaces.IAtom;
40import org.openscience.cdk.interfaces.IAtomContainer;
41import org.openscience.cdk.isomorphism.Mappings;
42import org.paukov.combinatorics3.Generator;
44import denoptim.constants.DENOPTIMConstants;
45import denoptim.exception.DENOPTIMException;
46import denoptim.fragmenter.BridgeHeadFindingRule;
47import denoptim.fragspace.APMapFinder;
48import denoptim.fragspace.FragmentSpace;
49import denoptim.fragspace.FragmentSpaceParameters;
50import denoptim.fragspace.GraphLinkFinder;
51import denoptim.fragspace.IdFragmentAndAP;
52import denoptim.graph.APClass;
53import denoptim.graph.APMapping;
54import denoptim.graph.AttachmentPoint;
55import denoptim.graph.DGraph;
56import denoptim.graph.Edge;
57import denoptim.graph.RelatedAPPair;
58import denoptim.graph.Ring;
59import denoptim.graph.SymmetricAPs;
60import denoptim.graph.SymmetricVertexes;
61import denoptim.graph.Template;
62import denoptim.graph.Template.ContractLevel;
63import denoptim.graph.Vertex;
64import denoptim.graph.Vertex.BBType;
65import denoptim.graph.rings.ChainLink;
66import denoptim.graph.rings.ClosableChain;
67import denoptim.graph.rings.PathSubGraph;
68import denoptim.graph.rings.RandomCombOfRingsIterator;
69import denoptim.graph.rings.RingClosingAttractor;
70import denoptim.graph.rings.RingClosureParameters;
71import denoptim.io.DenoptimIO;
72import denoptim.logging.CounterID;
73import denoptim.logging.Monitor;
74import denoptim.molecularmodeling.ThreeDimTreeBuilder;
75import denoptim.programs.RunTimeParameters.ParametersType;
76import denoptim.programs.denovo.GAParameters;
77import denoptim.programs.fragmenter.CuttingRule;
78import denoptim.programs.fragmenter.MatchedBond;
79import denoptim.utils.CrossoverType;
80import denoptim.utils.GraphUtils;
81import denoptim.utils.ManySMARTSQuery;
82import denoptim.utils.MutationType;
83import denoptim.utils.ObjectPair;
84import denoptim.utils.Randomizer;
110 int maxSizeXoverSubGraph)
116 List<Vertex[]> compatibleVrtxPairs =
new ArrayList<Vertex[]>();
117 for (
Edge eA : graphA.getEdgeList())
119 Vertex vA = eA.getTrgAP().getOwner();
124 for (
Edge eB : graphB.getEdgeList())
126 Vertex vB = eB.getTrgAP().getOwner();
135 compatibleVrtxPairs.add(pair);
143 ArrayList<XoverSite> sites =
new ArrayList<XoverSite>();
144 for (
Vertex[] pair : compatibleVrtxPairs)
152 List<Vertex> descendantsA =
new ArrayList<Vertex>();
154 List<Vertex> descendantsB =
new ArrayList<Vertex>();
169 List<Vertex> branchOnVA =
new ArrayList<Vertex>();
171 branchOnVA.addAll(descendantsA);
172 List<Vertex> branchOnVB =
new ArrayList<Vertex>();
174 branchOnVB.addAll(descendantsB);
188 List<Vertex[]> usablePairs =
new ArrayList<Vertex[]>();
203 for (
Vertex[] otherPair : compatibleVrtxPairs)
207 Vertex nextToEndOnA = otherPair[0];
208 Vertex nextToEndOnB = otherPair[1];
211 if (endOnA==
null || endOnB==
null)
215 if (!descendantsA.contains(endOnA) && endOnA!=vA
216 || !descendantsB.contains(endOnB) && endOnB!=vB)
237 if (!usablePairs.contains(pairOfEnds))
238 usablePairs.add(pairOfEnds);
242 TreeMap<String,List<Vertex[]>> sitesByBranchIdA =
243 new TreeMap<String,List<Vertex[]>>();
244 TreeMap<String,List<Vertex[]>> sitesByBranchIdB =
245 new TreeMap<String,List<Vertex[]>>();
246 for (
Vertex[] pp : usablePairs)
250 if (sitesByBranchIdA.containsKey(branchIdA))
252 sitesByBranchIdA.get(branchIdA).add(pp);
254 ArrayList<Vertex[]> lst =
new ArrayList<Vertex[]>();
256 sitesByBranchIdA.put(branchIdA, lst);
258 if (sitesByBranchIdB.containsKey(branchIdB))
260 sitesByBranchIdB.get(branchIdB).add(pp);
262 ArrayList<Vertex[]> lst =
new ArrayList<Vertex[]>();
264 sitesByBranchIdB.put(branchIdB, lst);
270 TreeMap<String,List<Vertex[]>> fewestBranchesSide =
null;
271 if (sitesByBranchIdA.size() <= sitesByBranchIdB.size())
272 fewestBranchesSide = sitesByBranchIdA;
274 fewestBranchesSide = sitesByBranchIdB;
277 for (List<
Vertex[]> val : fewestBranchesSide.values())
278 val.add(
new Vertex[]{
null,
null});
283 List<List<Vertex[]>> preCombsOfEnds = Generator.cartesianProduct(
284 fewestBranchesSide.values())
287 .collect(Collectors.<List<
Vertex[]>>toList());
291 List<List<Vertex[]>> combsOfEnds =
new ArrayList<List<Vertex[]>>();
292 for (List<
Vertex[]> comb : preCombsOfEnds)
294 List<Vertex[]> nullPurgedComb =
new ArrayList<Vertex[]>();
295 for (
Vertex[] inPair : comb)
297 if (inPair[0]!=
null && inPair[1]!=
null)
298 nullPurgedComb.add(inPair);
302 if (nullPurgedComb.size()>0)
303 combsOfEnds.add(nullPurgedComb);
326 for (
Vertex vA : graphA.getVertexList())
335 for (
Vertex vB : graphB.getVertexList())
346 maxSizeXoverSubGraph))
348 if (!sites.contains(xos))
371 List<
Vertex[]> cominationOfEnds,
376 if (cominationOfEnds.size()==0)
397 Generator.permutation(cominationOfEnds)
420 List<
Vertex[]> chosenSequenceOfEndpoints,
429 boolean exclude =
false;
430 for (
Vertex[] pairA : chosenSequenceOfEndpoints)
432 for (
Vertex[] pairB : chosenSequenceOfEndpoints)
437 if (pairA[0]==pairB[0] || pairA[1]==pairB[1])
449 List<Vertex> subGraphEndInA =
new ArrayList<Vertex>();
450 List<Vertex> subGraphEndInB =
new ArrayList<Vertex>();
451 List<Vertex> alreadyIncludedFromA =
new ArrayList<Vertex>();
452 List<Vertex> alreadyIncludedFromB =
new ArrayList<Vertex>();
453 for (
Vertex[] otherPair : chosenSequenceOfEndpoints)
455 Vertex endOnA = otherPair[0];
456 Vertex endOnB = otherPair[1];
459 if (alreadyIncludedFromA.contains(endOnA)
460 || alreadyIncludedFromB.contains(endOnB))
465 subGraphEndInA.add(endOnA);
466 subGraphEndInB.add(endOnB);
470 ArrayList<Vertex> subGraphA =
new ArrayList<Vertex>();
472 if (!subGraphEndInA.contains(vA))
475 ArrayList<Vertex> subGraphB =
new ArrayList<Vertex>();
477 if (!subGraphEndInB.contains(vB))
481 if (subGraphA.size()>1 && subGraphB.size()>1)
488 if (subGraphA.get(0).sameAs(subGraphB.get(0),
new StringBuilder()))
508 List<Vertex> subGraphA,
510 List<XoverSite> collector)
512 DGraph gOwnerA = subGraphA.get(0).getGraphOwner();
513 DGraph gOwnerB = subGraphB.get(0).getGraphOwner();
525 if (allAPsA.size() < needyAPsB.size()
526 || allAPsB.size() < needyAPsA.size())
543 && jacketTmplB==
null)
546 needyAPsB, xoverType);
547 if (!collector.contains(xos))
558 for (
Vertex v : subGraphA)
561 for (
Vertex v : subGraphB)
582 allAPsA, needyAPsA, allAPsB, needyAPsB,
590 subGraphB, needyAPsB, xoverType);
591 if (!collector.contains(xos))
644 DGraph graph = vertex.getGraphOwner();
692 DGraph graph = vertex.getGraphOwner();
727 return extendLink(vertex, chosenAPId, -1 , mnt, fragSpace);
764 +vertex+
" has no edge user.");
769 +
"vertex (AP "+chosenAPId+
" of vertex "+vertex+
").");
771 return extendLink(e, chosenNewVrtxId, mnt, fragSpace);
810 LinkedHashMap<AttachmentPoint,Integer> apMap =
811 new LinkedHashMap<AttachmentPoint,Integer>();
812 for (Entry<AttachmentPoint, AttachmentPoint> e :
815 apMap.put(e.getKey(), e.getValue().getIndexInOwner());
818 DGraph graph = edge.getSrcAP().getOwner().getGraphOwner();
855 boolean force,
int chosenVrtxIdx,
int chosenApId,
858 DGraph g = vertex.getGraphOwner();
861 Edge e = vertex.getEdgeToParent();
864 String msg =
"Program Bug in substituteFragment: Unable to locate "
865 +
"parent edge for vertex "+vertex+
" in graph "+g;
866 settings.getLogger().log(Level.SEVERE, msg);
881 return extendGraph(parentVrt,
false,symmetry,force,chosenVrtxIdx,
882 chosenApId, settings);
898 long vid = vertex.getVertexId();
899 DGraph molGraph = vertex.getGraphOwner();
903 List<Vertex> toRemove =
new ArrayList<Vertex>();
938 long vid = vertex.getVertexId();
939 DGraph molGraph = vertex.getGraphOwner();
943 List<Vertex> toRemove =
new ArrayList<Vertex>();
947 if (!v.getMutationTypes(
new ArrayList<MutationType>())
994 vertex.getGraphOwner().removeCappingGroups();
996 List<AttachmentPoint> freeeAPs = vertex.getFreeAPThroughout();
997 if (freeeAPs.size()==0)
1002 DGraph originalGraph = vertex.getGraphOwner();
1009 List<Ring> setOfRingsOnTmpGraph =
null;
1012 APClass apc = srcAP.getAPClass();
1015 if (!fragSpace.getRCCompatibilityMatrix().containsKey(apc))
1019 List<APClass> rcTrgAPCs = fragSpace.getRCCompatibilityMatrix().get(
1023 Vertex rcvOnSrcAP =
null;
1024 List<Vertex> candidateRCVs = fragSpace.getRCVsForAPClass(apc);
1025 boolean rcvIsChosenArbitrarily =
false;
1026 if (candidateRCVs.size()>0)
1030 rcvIsChosenArbitrarily =
true;
1031 rcvOnSrcAP = fragSpace.getPolarizedRCV(
true);
1035 List<Vertex> rcvAddedToGraph =
new ArrayList<Vertex>();
1037 rcvAddedToGraph.add(rcvOnSrcAP);
1042 for (
int i=0; i<20; i++)
1045 List<AttachmentPoint> apsToTry =
1047 int numberOfAttempts = apsToTry.size();
1049 for (
int iap=0; iap<numberOfAttempts; iap++)
1052 if (rcTrgAPCs.contains(candidate.
getAPClass()))
1057 apsToTry.remove(trgAP);
1066 Vertex rcvOnTrgAP =
null;
1067 if (rcvIsChosenArbitrarily)
1069 rcvOnTrgAP = fragSpace.getPolarizedRCV(
false);
1071 List<Vertex> candRCVs = fragSpace.getRCVsForAPClass(
1073 if (candRCVs.size()>0)
1077 List<Vertex> keptRCVs =
new ArrayList<Vertex>();
1078 for (
Vertex rcv : candRCVs)
1080 if (requiredAPC.
equals(rcv.getAP(0).getAPClass()))
1084 if (rcvOnTrgAP==
null)
1099 rcvAddedToGraph.add(rcvOnTrgAP);
1101 if (rcvAddedToGraph.size() < 2)
1110 settings.getLogger(), rng);
1116 settings.getMaxRingsAddedByMutation(),
1117 fragSpace, rcParams);
1123 setOfRingsOnTmpGraph = rCombIter.
next();
1126 if (setOfRingsOnTmpGraph.size()>0)
1130 for (
Vertex toRemove : rcvAddedToGraph)
1133 if (setOfRingsOnTmpGraph==
null || setOfRingsOnTmpGraph.size()==0)
1140 boolean done =
false;
1141 for (
Ring rOnTmp : setOfRingsOnTmpGraph)
1145 tmpGraph.
indexOf(rOnTmp.getHeadVertex().getParent()));
1147 rOnTmp.getHeadVertex().getEdgeToParent().getSrcAP()
1148 .getIndexInOwner());
1154 headRCV = rOnTmp.getHeadVertex().
clone();
1162 tmpGraph.
indexOf(rOnTmp.getTailVertex().getParent()));
1164 rOnTmp.getTailVertex().getEdgeToParent().getSrcAP()
1165 .getIndexInOwner());
1171 tailRCV = rOnTmp.getTailVertex().
clone();
1178 originalGraph.
addRing(headRCV, tailRCV);
1183 vertex.getGraphOwner().addCappingGroups(fragSpace);
1220 vertex.getGraphOwner().removeCappingGroups();
1222 List<AttachmentPoint> freeAPs = vertex.getFreeAPThroughout();
1223 if (freeAPs.size()==0)
1229 DGraph graph = vertex.getGraphOwner();
1234 List<List<RelatedAPPair>> candidatePairsSets =
1239 rng.
nextBoolean(settings.getSymmetryProbability()),
1240 settings.getLogger(), rng);
1241 if (candidatePairsSets.size()==0)
1251 List<List<RelatedAPPair>> szBiasedCandidatePairsSets =
1252 new ArrayList<List<RelatedAPPair>>();
1253 for (List<RelatedAPPair> pairSet : candidatePairsSets)
1256 pairSet.get(0).property;
1260 for (
int i=0; i<allowedBridgeLengths.length; i++)
1262 int allowedBridgeLength = allowedBridgeLengths[i];
1263 int possibleRingSize = allowedBridgeLength
1264 + existingBridgeLength;
1274 for (
int z=0; z<weight; z++)
1276 szBiasedCandidatePairsSets.add(pairSet);
1280 if (szBiasedCandidatePairsSets.size()==0)
1288 szBiasedCandidatePairsSets);
1293 String elsInHalfFrag = chosenPairsSet.get(0).propID.substring(0,1);
1294 if (elsInHalfFrag.matches(
"[a-zA-Z]"))
1295 elsInHalfFrag =
"0";
1296 boolean newRingIsAromatic =
true;
1297 String elInIncomingFrag =
"0el";
1298 switch (elsInHalfFrag)
1303 elInIncomingFrag =
"0el";
1304 newRingIsAromatic =
false;
1310 elInIncomingFrag =
"4el";
1316 elInIncomingFrag =
"3el";
1321 elInIncomingFrag =
"2el";
1327 elInIncomingFrag =
"1el";
1330 throw new Error(
"Unknown number of pi-electrons in fragment to "
1331 +
"be used for ring fusion operation.");
1337 chosenPairsSet.get(0).property;
1339 List<Vertex> usableBridges =
null;
1340 if (newRingIsAromatic)
1350 chosenPairsSet.get(0).apA.getAPClass(),
1351 chosenPairsSet.get(0).apB.getAPClass(),
1356 if (usableBridges.size()==0)
1365 List<Vertex> szBiasedUsableBridges =
new ArrayList<Vertex>();
1366 for (
Vertex candidateBridge : usableBridges)
1368 int thisBridgeLength = (int) candidateBridge.getProperty(
1371 int resultingRingSize = existingBridgeLength + thisBridgeLength;
1381 for (
int z=0; z<weigth; z++)
1383 szBiasedUsableBridges.add(candidateBridge);
1387 if (szBiasedUsableBridges.size()==0)
1396 List<AttachmentPoint> apsInFusion =
new ArrayList<AttachmentPoint>();
1397 int[] idApOnBridge =
new int[2];
1398 if (newRingIsAromatic)
1404 idApOnBridge[0] = apsInFusion.get(0).getIndexInOwner();
1405 idApOnBridge[1] = apsInFusion.get(1).getIndexInOwner();
1407 idApOnBridge[0] = apsInFusion.get(1).getIndexInOwner();
1408 idApOnBridge[1] = apsInFusion.get(0).getIndexInOwner();
1411 idApOnBridge[0] = Integer.parseInt(incomingVertex.
getProperty(
1413 idApOnBridge[1] = Integer.parseInt(incomingVertex.
getProperty(
1417 boolean done =
false;
1428 Vertex rcvBridge = fragSpace.getPolarizedRCV(
true);
1430 rcvBridge.
getAP(0));
1432 Vertex rcvTail = fragSpace.getPolarizedRCV(
false);
1434 graph.
addRing(rcvBridge, rcvTail);
1439 vertex.getGraphOwner().addCappingGroups(fragSpace);
1462 boolean symmetryOnAps,
1466 return extendGraph(curVertex, extend, symmetryOnAps,
false, -1, -1,
1497 boolean symmetryOnAps,
1520 boolean status =
false;
1523 if (!curVrtx.hasFreeAP())
1528 DGraph molGraph = curVrtx.getGraphOwner();
1529 int lvl = molGraph.
getLevel(curVrtx);
1531 ArrayList<Long> addedVertices =
new ArrayList<>();
1533 List<AttachmentPoint> lstDaps = curVrtx.getAttachmentPoints();
1534 List<AttachmentPoint> toDoAPs =
new ArrayList<AttachmentPoint>();
1535 toDoAPs.addAll(lstDaps);
1536 for (
int i=0; i<lstDaps.size(); i++)
1557 boolean allowOnlyRingClosure =
false;
1569 double extendGraphProb = molSizeProb * byLevelProb * crowdingProb;
1570 boolean fgrow = settings.getRandomizer().nextBoolean(
1575 && settings.getRandomizer().nextBoolean(byLevelProb
1578 allowOnlyRingClosure =
true;
1586 if (!allowOnlyRingClosure
1591 apId, molGraph, addedVertices, settings);
1600 if (allowOnlyRingClosure)
1625 if ((curVrtx.getGraphOwner().getHeavyAtomsCount() +
1635 settings.getSymmetryProbability(),
1638 if (curVrtx.getSymmetricAPs(ap).size()!=0
1639 && (cpOnSymAPs || symmetryOnAps)
1640 && !allowOnlyRingClosure)
1642 symAPs.addAll(curVrtx.getSymmetricAPs(ap));
1645 boolean allOnSameSrc =
true;
1648 if (!symAP.hasSameSrcAtom(ap))
1650 allOnSameSrc =
false;
1667 crowdedness = crowdedness + 1;
1671 double shot = settings.getRandomizer().nextDouble();
1680 crowdedness, settings);
1682 if (shot > crowdProb)
1686 crowdedness = crowdedness + 1;
1706 symVerts.
add(curVrtx);
1710 if ((curVrtx.getGraphOwner().getHeavyAtomsCount() +
1712 * symAPs.size()) > maxHeavyAtoms)
1718 List<AttachmentPoint> allAPsFromSymVerts =
new ArrayList<>();
1719 for (
Vertex symVrt : symVerts)
1724 apOnVrt.getIndexInOwner());
1728 if (apOnVrt.sameAs(apOnSymVrt)
1730 && !symAPs.contains(apOnSymVrt))
1732 allAPsFromSymVerts.add(apOnSymVrt);
1736 symAPs.addAll(allAPsFromSymVerts);
1744 if (!symAP.isAvailable())
1760 addedVertices.add(newVrtId);
1761 newSymSetOfVertices.
add(fragVertex);
1765 if (newSymSetOfVertices.size() > 1)
1774 for (
int i=0; i<addedVertices.size(); i++)
1776 long vid = addedVertices.get(i);
1782 if (addedVertices.size() > 0)
1827 int dapidx,
int chosenVrtxIdx,
int chosenApId,
1830 List<AttachmentPoint> lstDaps = curVertex.getAttachmentPoints();
1836 if (!fragSpace.useAPclassBasedApproach())
1838 int fid = fragSpace.getRandomizer().nextInt(
1839 fragSpace.getFragmentLibrary().size());
1844 List<IdFragmentAndAP> candidates =
1845 fragSpace.getFragAPsCompatibleWithClass(
1847 if (candidates.size() > 0)
1849 if (chosenVrtxIdx>-1 && chosenApId>-1)
1855 res = fragSpace.getRandomizer().randomlyChooseOne(candidates);
1881 List<Vertex> rcvs = fragSpace.getRCVs();
1883 if (!fragSpace.useAPclassBasedApproach())
1887 List<Vertex> candidates = fragSpace.getRCVsForAPClass(
1889 if (candidates.size() > 0)
1909 boolean res =
false;
1919 curVertex, dapidx, molGraph);
1930 int numCands = lscFfCc.size();
1933 int chosenId = settings.getRandomizer().nextInt(numCands);
1935 ArrayList<Integer> newFragIds = chosenFfCc.
getFragIDs();
1936 int molIdNewFrag = newFragIds.get(0);
1938 int dapNewFrag = newFragIds.get(2);
1939 if (molIdNewFrag != -1)
1943 newvid, molIdNewFrag, typeNewFrag,
1946 molGraph.appendVertexOnAP(curVertex.getAP(dapidx),
1947 newVrtx.
getAP(dapNewFrag));
1951 addedVertices.add(newvid);
1953 molGraph.getClosableChains().removeAll(
1955 APClass apc = curVertex.getAttachmentPoints().get(
1956 dapidx).getAPClass();
1959 settings.getSymmetryProbability(),
1968 String msg =
"BUG: Incorrect vertex num. Contact author.";
1969 settings.getLogger().log(Level.SEVERE, msg);
2045 ArrayList<FragForClosabChains> lstChosenFfCc =
2046 new ArrayList<FragForClosabChains>();
2048 if (molGraph.getClosableChains().size() == 0)
2050 return lstChosenFfCc;
2057 int posInCc = cc.involvesVertex(curVertex);
2072 if (cc.getSize() > (posInCc+1))
2075 ChainLink nextChainLink = cc.getLink(posInCc+1);
2076 nfid = nextChainLink.
getIdx();
2088 if ((posInCc-1) >= 0)
2091 ChainLink nextChainLink = cc.getLink(posInCc-1);
2092 nfid = nextChainLink.
getIdx();
2103 ArrayList<Integer> eligibleFrgId =
new ArrayList<Integer>();
2104 eligibleFrgId.add(nfid);
2105 eligibleFrgId.add(nfty.
toOldInt());
2106 eligibleFrgId.add(nfap);
2107 boolean found =
false;
2110 int fidA = ffcc.getFragIDs().get(0);
2112 int fapA = ffcc.getFragIDs().get(2);
2113 if (nfid==fidA && nfty==ftyA && nfap==fapA)
2116 ffcc.getCompatibleCC().add(cc);
2120 ffcc.getIncompatibleCC().add(cc);
2125 ArrayList<ClosableChain> compatChains =
2126 new ArrayList<ClosableChain>();
2127 ArrayList<ClosableChain> incompatChains =
2128 new ArrayList<ClosableChain>();
2131 incompatChains.addAll(otherFfCc.getCompatibleCC());
2139 lstChosenFfCc.add(newChosenCc);
2146 Edge edge = molGraph.getEdgeWithParent(
2147 curVertex.getVertexId());
2154 int posInCc = cc.involvesVertexAndAP(curVertex, dapidx, chidAp);
2168 List<Integer> altertnativeDirections =
new ArrayList<>();
2169 altertnativeDirections.add(-1);
2170 altertnativeDirections.add(+1);
2171 for (
int altDir : altertnativeDirections)
2173 ChainLink parentLink = cc.getLink(posInCc + altDir);
2174 int pLnkId = parentLink.
getIdx();
2187 if (pLnkId==prntId && pLnkTyp==prntTyp && pLnkAp == prntAp &&
2190 if (cc.getSize() > (posInCc+altDir)
2191 && (posInCc+altDir) >= 0)
2193 ChainLink nextChainLink = cc.getLink(posInCc + altDir);
2194 nfid = nextChainLink.
getIdx();
2211 ArrayList<Integer> eligibleFrgId =
new ArrayList<Integer>();
2212 eligibleFrgId.add(nfid);
2213 eligibleFrgId.add(nfty.
toOldInt());
2214 eligibleFrgId.add(nfap);
2215 boolean found =
false;
2218 int fidA = ffcc.getFragIDs().get(0);
2220 int fapA = ffcc.getFragIDs().get(2);
2221 if (nfid==fidA && nfty==ftyA && nfap==fapA)
2224 ffcc.getCompatibleCC().add(cc);
2228 ffcc.getIncompatibleCC().add(cc);
2233 ArrayList<ClosableChain> compatChains =
2234 new ArrayList<ClosableChain>();
2235 ArrayList<ClosableChain> incompatChains =
2236 new ArrayList<ClosableChain>();
2239 incompatChains.addAll(otherFfCc.getCompatibleCC());
2246 lstChosenFfCc.add(newChosenCc);
2250 return lstChosenFfCc;
2268 DGraph gA = site.getA().get(0).getGraphOwner();
2269 DGraph gB = site.getB().get(0).getGraphOwner();
2272 List<AttachmentPoint> allAPsOnA = gA.
getSubgraphAPs(site.getA());
2273 List<AttachmentPoint> allAPsOnB = gB.
getSubgraphAPs(site.getB());
2277 List<AttachmentPoint> needyAPsOnA = site.getAPsNeedingMappingA();
2278 List<AttachmentPoint> needyAPsOnB = site.getAPsNeedingMappingB();
2285 if (!ap.isSrcInUserThroughout())
2294 if (!ap.isSrcInUserThroughout())
2301 if (apToParentA==
null || apToParentB==
null)
2304 +
"point connecting a subgraph to the rest of the graph. "
2305 +
"This violates assumption that crossover does not "
2306 +
"involve scaffold or vertexes without parent.");
2319 fixedRootAPs.put(apToParentA, apToParentB);
2325 allAPsOnA, needyAPsOnA,
2326 allAPsOnB, needyAPsOnB, fixedRootAPs,
2345 LinkedHashMap<AttachmentPoint,AttachmentPoint>
2346 apMapA =
new LinkedHashMap<AttachmentPoint,AttachmentPoint>();
2347 LinkedHashMap<AttachmentPoint,AttachmentPoint>
2348 apMapB =
new LinkedHashMap<AttachmentPoint,AttachmentPoint>();
2363 apMapA.put(apOnA, apOnSubGraphB);
2364 apMapB.put(apOnB, apOnSubGraphA);
2391 double symmetryProbability,
Randomizer randomizer)
2425 List<Vertex> mutable = graph.getMutableSites(
2426 settings.getExcludedMutationTypes());
2434 Vertex vA = site.apA.getOwner();
2435 Vertex vB = site.apB.getOwner();
2436 if (!mutable.contains(vA))
2438 if (!mutable.contains(vB))
2443 if (mutable.size() == 0)
2446 String msg =
"Graph has no mutable site. Mutation aborted.";
2447 settings.getLogger().info(msg);
2450 boolean doneMutation =
true;
2452 settings.getMultiSiteMutationWeights(),
2453 settings.getRandomizer().nextDouble());
2454 for (
int i=0; i<numberOfMutations; i++)
2458 mutable = graph.getMutableSites(
2459 settings.getExcludedMutationTypes());
2462 Vertex v = settings.getRandomizer().randomlyChooseOne(mutable);
2467 return doneMutation;
2488 List<MutationType> mTypes = vertex.getMutationTypes(
2489 settings.getExcludedMutationTypes());
2490 if (mTypes.size() == 0)
2494 MutationType mType = settings.getRandomizer().randomlyChooseOne(mTypes);
2516 int pos = vertex.getGraphOwner().
indexOf(vertex);
2520 }
catch (IllegalArgumentException|NullPointerException e)
2522 String debugFile =
"failedMutation_" + mType
2523 +
"_" + vertex.getVertexId() +
"(" + pos +
")_"
2524 + settings.timeStamp +
".sdf";
2526 settings.getLogger(), settings.getRandomizer());
2527 settings.getLogger().warning(
"Fatal exception while performing "
2528 +
"mutation. See file '" + debugFile +
"' to reproduce the "
2573 DGraph graph = vertex.getGraphOwner();
2577 settings.getLogger().info(
"Vertex has no owner - "
2578 +
"Mutation aborted");
2581 if (!vertex.getMutationTypes(settings.getExcludedMutationTypes())
2585 settings.getLogger().info(
"Vertex does not allow mutation type "
2586 +
"'" + mType +
"' - Mutation aborted");
2591 int positionOfVertex = graph.
indexOf(vertex);
2598 +
" (" + positionOfVertex +
")";
2601 boolean done =
false;
2605 done =
rebuildBranch(vertex, force, chosenVrtxIdx, chosenApId,
2638 List<Integer> candidates =
new ArrayList<Integer>();
2641 candidates.add(c.getEdgeToParent().getSrcAP()
2642 .getIndexInOwner());
2644 if (candidates.size() == 0)
2650 chosenApId = settings.getRandomizer().randomlyChooseOne(
2653 done =
extendLink(vertex, chosenApId, chosenVrtxIdx, mnt,
2675 vertex.getGraphOwner().removeCappingGroupsOn(vertex);
2676 done =
extendGraph(vertex,
false,
false, force, chosenVrtxIdx,
2677 chosenApId, settings);
2689 String msg =
"Mutation '" + mType.toString() +
"' on vertex "
2690 + vertex.toString() +
" (position " + positionOfVertex
2691 +
" in graph " + graphId+
"): ";
2703 msg = msg +
"unsuccessful";
2705 settings.getLogger().info(msg);
General set of constants used in DENOPTIM.
static final String VRTPROPBRIDGELENGTH
Name of Vertex property used to record how long a ring-closing bridge is.
static final String VRTPROPBRIDGEEND_B
Name of Vertex property used to record which AP is selected for bridge formation on side 'B'.
static final Object STOREDVID
Key of the property remembering vertex IDs.
static final String VRTPROPBRIDGEEND_A
Name of Vertex property used to record which AP is selected for bridge formation on side 'A'.
SMARTS-based rules to identify potential bridge head atoms for ring fusion operations.
int getExistingBridgeLength()
int[] getAllowedBridgeLength()
An utility class to encapsulate the search for an AttachmentPoint-AttachmentPoint mapping.
APMapping getChosenAPMapping()
Returns the AttachmentPoint-AttachmentPoint mapping chosen among the possible mappings.
boolean foundMapping()
Returns true if any mapping has been found.
Class defining a space of building blocks.
boolean imposeSymmetryOnAPsOfClass(APClass apClass)
Checks if the symmetry settings impose use of symmetry on attachment points of the given AP class.
Parameters defining the fragment space.
FragmentSpace getFragmentSpace()
An utility class to encapsulate the search for vertexes that satisfy constraints.
Vertex getChosenAlternativeLink()
Returns the vertex chosen as alternative.
APMapping getChosenAPMapping()
Returns the AP mapping that allows the usage of the vertex chosen either as alternative to be install...
boolean foundAlternativeLink()
Data structure containing information that identifies a single AP of a vertex/fragment.
Helper methods for the genetic algorithm.
static double getGrowthByLevelProbability(int level, GAParameters settings)
Calculates the probability of adding a fragment to the given level.
static int chooseNumberOfSitesToMutate(double[] multiSiteMutationProb, double hit)
Takes a decision on how many sites to mutate on a candidate.
static List< Vertex > getUsableAliphaticBridges(APClass apcA, APClass apcB, int[] allowedLengths, FragmentSpace fragSpace)
Finds all vertexes that can be used as aliphatic bridge.
static double getCrowdingProbability(AttachmentPoint ap, GAParameters settings)
Calculated the probability of using and attachment point rooted on an atom that is holding other atta...
static List< List< RelatedAPPair > > searchRingFusionSites(DGraph graph, GAParameters gaParams)
static int getCrowdedness(AttachmentPoint ap)
Calculate the current crowdedness of the given attachment point.
static List< Vertex > getUsableAromaticBridges(String elInIncomingFrag, int[] allowedLengths, FragmentSpace fragSpace)
Finds all vertexes that can be used as aromatic bridge, i.e., can be used to create an aromatic ring ...
static double getMolSizeProbability(DGraph graph, GAParameters settings)
Calculated the probability of extending a graph based on the current size of the molecular representa...
Private class representing a selected closable chain of fragments.
ArrayList< ClosableChain > incompatChains
FragForClosabChains(ArrayList< ClosableChain > compatChains, ArrayList< ClosableChain > incompatChains, ArrayList< Integer > fragIds)
ArrayList< Integer > fragIds
ArrayList< ClosableChain > compatChains
ArrayList< ClosableChain > getCompatibleCC()
ArrayList< ClosableChain > getIncompatibleCC()
void addCompatibleCC(ClosableChain icc)
ArrayList< Integer > getFragIDs()
Collection of operators meant to alter graphs and associated utilities.
static IdFragmentAndAP getFrgApForSrcAp(Vertex curVertex, int dapidx, FragmentSpace fragSpace)
Select a compatible fragment for the given attachment point.
static boolean performMutation(Vertex vertex, MutationType mType, boolean force, int chosenVrtxIdx, int chosenApId, Monitor mnt, GAParameters settings)
Mutates the given vertex according to the given mutation type, if possible.
static boolean extendGraph(Vertex curVrtx, boolean extend, boolean symmetryOnAps, boolean force, int chosenVrtxIdx, int chosenApId, GAParameters settings)
function that will keep extending the graph.
static boolean extendLink(Vertex vertex, int chosenAPId, Monitor mnt, FragmentSpace fragSpace)
Inserts a vertex in between two connected vertexes that are identified by the one vertex holding the ...
static List< XoverSite > locateCompatibleXOverPoints(DGraph graphA, DGraph graphB, FragmentSpace fragSpace, int maxSizeXoverSubGraph)
Identify crossover sites, i.e., subgraphs that can be swapped between two graphs (i....
static boolean applySymmetry(boolean apclassImposed, double symmetryProbability, Randomizer randomizer)
Decides whether to apply constitutional symmetry or not.
static void checkAndAddXoverSites(FragmentSpace fragSpace, List< Vertex > subGraphA, List< Vertex > subGraphB, CrossoverType xoverType, List< XoverSite > collector)
Here we check that the given subgraphs have a mapping that allows to swap them, and full fill any oth...
static boolean performMutation(Vertex vertex, MutationType mType, Monitor mnt, GAParameters settings)
Mutates the given vertex according to the given mutation type, if possible.
static boolean substituteLink(Vertex vertex, int chosenVrtxIdx, Monitor mnt, FragmentSpace fragSpace)
Substitutes a vertex while keeping its surrounding.
static boolean isCrossoverPossible(Edge eA, Edge eB, FragmentSpace fragSpace)
Evaluate AP class-compatibility of a pair of edges with respect to crossover.
static void processCombinationOfEndPoints(Vertex[] pair, List< Vertex[]> cominationOfEnds, List< XoverSite > collector, FragmentSpace fragSpace)
Given a pair of seed vertexes that define where a pair of subgraphs stars in the corresponding origin...
static boolean attachFragmentInClosableChain(Vertex curVertex, int dapidx, DGraph molGraph, ArrayList< Long > addedVertices, GAParameters settings)
static void processPermutationOfEndPoints(Vertex[] pair, List< Vertex[]> chosenSequenceOfEndpoints, List< XoverSite > collector, FragmentSpace fragSpace)
Given a pair of seed vertexes that define where a pair of subgraphs stars in the corresponding origin...
static boolean extendGraph(Vertex curVertex, boolean extend, boolean symmetryOnAps, GAParameters settings)
function that will keep extending the graph according to the growth/substitution probability.
static ArrayList< FragForClosabChains > getFragmentForClosableChain(Vertex curVertex, int dapidx, DGraph molGraph)
Method to select fragments that increase the likeliness of generating closable chains.
static boolean performMutation(Vertex vertex, Monitor mnt, GAParameters settings)
Tries to do mutate the given vertex.
static IdFragmentAndAP getFrgApForSrcAp(Vertex curVertex, int dapidx, int chosenVrtxIdx, int chosenApId, FragmentSpace fragSpace)
Select a compatible fragment for the given attachment point.
static IdFragmentAndAP getRCVForSrcAp(Vertex curVertex, int dapidx, FragmentSpace fragSpace)
Select a compatible ring-closing vertex for the given attachment point.
static boolean deleteFragment(Vertex vertex)
Deletion mutation removes the vertex and also the symmetric partners on its parent.
static boolean rebuildBranch(Vertex vertex, boolean force, int chosenVrtxIdx, int chosenApId, GAParameters settings)
Substitutes a vertex and any child branch.
static boolean addFusedRing(Vertex vertex, Monitor mnt, boolean force, FragmentSpace fragSpace, GAParameters settings)
Tries to add a fused ring using a pair of free APs, one of which on the given vertex.
static boolean deleteLink(Vertex vertex, int chosenVrtxIdx, Monitor mnt, FragmentSpace fragSpace)
Removes a vertex while merging as many of the child branches into the parent vertex.
static boolean extendLink(Vertex vertex, int chosenAPId, int chosenNewVrtxId, Monitor mnt, FragmentSpace fragSpace)
Inserts a vertex in between two connected vertexes that are identified by the one vertex holding the ...
static boolean performMutation(DGraph graph, Monitor mnt, GAParameters settings)
Tries to do mutate the given graph.
static boolean addRing(Vertex vertex, Monitor mnt, boolean force, FragmentSpace fragSpace, GAParameters settings)
Tries to use any free AP of the given vertex to close ring in the graph by adding a chord.
static boolean extendLink(Edge edge, int chosenBBIdx, Monitor mnt, FragmentSpace fragSpace)
Replace an edge with two edges with a new vertex in between, thus inserting a vertex in between two d...
static boolean deleteChain(Vertex vertex, Monitor mnt, FragmentSpace fragSpace)
Deletes the given vertex and all other vertexes that are not connected to more than 2 non-capping gro...
static boolean performCrossover(XoverSite site, FragmentSpace fragSpace)
Performs the crossover that swaps the two subgraphs defining the given XoverSite.
This class collects the data identifying the subgraphs that would be swapped by a crossover event.
boolean isCPMapCompatibleWith(APClass other, FragmentSpace fragSpace)
Check compatibility as defined in the compatibility matrix considering this AP as source and the othe...
Class representing a mapping between attachment points (APs).
LinkedHashMap< Integer, Integer > toIntMappig()
Produces an index-based version of this mapping where each index represents the attachment point as i...
An attachment point (AP) is a possibility to attach a Vertex onto the vertex holding the AP (i....
AttachmentPoint getLinkedAP()
Gets the attachment point (AP) that is connected to this AP via the edge user.
APClass getAPClass()
Returns the Attachment Point class.
boolean isAvailable()
Check availability of this attachment point.
Edge getEdgeUserThroughout()
Gets the edge that is using this AP, or null if no edge is using this AP.
Container for the list of vertices and the edges that connect them.
boolean removeVertexAndWeld(Vertex vertex, FragmentSpace fragSpace)
Remove a given vertex belonging to this graph and re-connects the resulting graph branches as much as...
boolean replaceVertex(Vertex vertex, int bbId, BBType bbt, LinkedHashMap< Integer, Integer > apIdMap, FragmentSpace fragSpace)
Replaced a given vertex belonging to this graph with a new vertex generated specifically for this pur...
Vertex getVertexWithId(long vid)
Searches for a vertex with the given identifier.
DGraph extractSubgraph(int index)
Creates a new graph that corresponds to the subgraph of this graph when exploring the spanning tree f...
boolean isIsomorphicTo(DGraph other)
Checks if this graph is "DENOPTIM-isomorphic" to the other one given.
ArrayList< Ring > getRingsInvolvingVertex(Vertex v)
Returns the list of rings that include the given vertex in their fundamental cycle.
void getChildrenTree(Vertex vertex, List< Vertex > children)
Gets all the children of the current vertex recursively.
void removeVertex(Vertex vertex)
Remove a vertex from this graph.
int indexOf(Vertex v)
Returns the index of a vertex in the list of vertices of this graph.
Vertex getVertexAtPosition(int pos)
Returns the vertex that is in the given position of the list of vertices belonging to this graph.
Vertex getDeepestAmongThese(List< Vertex > list)
Identify the oldest ancestor (i.e., most great grandparent) in the given collection.
String getBranchIdOfVertexAsStr(Vertex v)
Returns the branch identifier as a literal string.
List< AttachmentPoint > getAvailableAPsThroughout()
Returns the list of attachment points contained in this graph that are available throughout the templ...
void appendVertexOnAP(AttachmentPoint srcAP, AttachmentPoint trgAP)
Append a vertex to this graph: adds the new vertex to the list of vertices belonging to the graph,...
List< AttachmentPoint > getInterfaceAPs(List< Vertex > subGraphB)
Searches for all AttachmentPoints that represent the interface between a subgraph,...
DGraph clone()
Returns almost "deep-copy" of this graph.
boolean replaceSubGraph(List< Vertex > subGrpVrtxs, DGraph incomingGraph, LinkedHashMap< AttachmentPoint, AttachmentPoint > apMap, FragmentSpace fragSpace)
Replaced the subgraph represented by a given collection of vertices that belong to this graph.
int getLevel(Vertex v)
Calculates the level of a vertex in this graph.
void addSymmetricSetOfVertices(SymmetricVertexes symSet)
Adds a symmetric set of vertices to this graph.
SymmetricVertexes getSymSetForVertex(Vertex v)
Returns the set of vertexes symmetric to the given one.
List< AttachmentPoint > getSubgraphAPs(List< Vertex > subGraphB)
Searches for all AttachmentPoints that are owned by vertices in a subgraph but either available or us...
boolean hasSymmetryInvolvingVertex(Vertex v)
boolean insertVertex(Edge edge, int bbId, BBType bbt, LinkedHashMap< AttachmentPoint, Integer > apMap, FragmentSpace fragSpace)
Inserts a given vertex in between two vertices connected by the given edge.
void getChildTreeLimited(Vertex vertex, List< Vertex > children, boolean stopBeforeRCVs)
Gets all the children of the current vertex recursively.
boolean removeBranchStartingAt(Vertex v, boolean symmetry)
Deletes the branch, i.e., the specified vertex and its children.
Template getTemplateJacket()
boolean removeChainUpToBranching(Vertex v, FragmentSpace fragSpace)
Mutates the graph by removing the chain where a given vertex is located up to the first branching (i....
void setLocalMsg(String msg)
boolean isIsostructuralTo(DGraph other)
Checks if this graph is "DENOPTIM-isostructural" to the other one given.
This class represents the edge between two vertices.
AttachmentPoint getTrgAP()
AttachmentPoint getSrcAP()
This class represents the closure of a ring in a spanning tree.
A collection of AttachmentPoints that are related by a relation that we call "symmetry",...
boolean add(T item)
Adds an item to this list, if not already present.
A collection of Vertexs that are related by a relation that we call "symmetry", even though this clas...
ContractLevel getContractLevel()
Returns the contract level of this template, i.e., to what extent the content of this template can be...
void clearIAtomContainer()
Removes the molecular representation.
A vertex is a data structure that has an identity and holds a list of AttachmentPoints.
abstract Vertex clone()
Returns a deep-copy of this vertex.
List< AttachmentPoint > getAPsWithAPClassStartingWith(String root)
Finds only APs that have APClass starting with the given string.
int getBuildingBlockId()
Returns the index of the building block that should correspond to the position of the building block ...
Edge getEdgeToParent()
Looks into the edges that use any of the APs that belong to this vertex and returns the edge that has...
void setVertexId(long vertexId2)
Vertex getParent()
Looks into the edges that use any of the APs that belong to this vertex and returns the vertex which ...
Vertex.BBType getBuildingBlockType()
DGraph getGraphOwner()
Returns the graph this vertex belongs to or null.
ArrayList< Vertex > getChilddren()
Looks into the edges that use any of the APs that belong to this vertex and returns the list of verti...
Object getProperty(Object property)
abstract int getHeavyAtomsCount()
ArrayList< AttachmentPoint > getFreeAPThroughout()
Gets attachment points that are availability throughout the graph level, i.e., checks also across the...
AttachmentPoint getAP(int i)
Get attachment point i on this vertex.
static Vertex newVertexFromLibrary(int bbId, Vertex.BBType bbt, FragmentSpace fragSpace)
Builds a new molecular fragment kind of vertex.
ChainLink represents a vertex in a closable chain.
Vertex.BBType getFragType()
ClosableChain represents a chain of fragments (chain links) that is closable (or candidate closable).
This object represents a path in a DGraph.
int getPathLength()
Returns the length of the list of edges involved in this path.
List< Vertex > getVertecesPath()
Returns the list of verteces involved.
A class for iterating over sets of ring combinations generated by considering any constrain and setti...
The RingClosingAttractor represent the available valence/connection that allows to close a ring.
static final Map< APClass, APClass > RCAAPCMAP
Recognized APClasses on RingClosingAttractor and compatible types.
Parameters and setting related to handling ring closures.
void allowRingClosures(boolean value)
List< Integer > getRingSizeBias()
boolean selectFragmentsFromClosableChains()
Utility methods for input/output.
static void writeGraphToSDF(File file, DGraph graph, boolean append, boolean make3D, Logger logger, Randomizer randomizer)
Writes the graph to SDF file.
A collection of counters user to count actions taken by the evolutionary algorithm.
Tool to build build three-dimensional (3D) tree-like molecular structures from DGraph.
void setAlignBBsIn3D(boolean align)
Sets the flag that controls whether building blocks have to be aligned according to the AP vectors or...
IAtomContainer convertGraphTo3DAtomContainer(DGraph graph)
Created a three-dimensional molecular representation from a given DGraph.
RunTimeParameters getParameters(ParametersType type)
Randomizer getRandomizer()
Returns the current program-specific randomizer.
Parameters for genetic algorithm.
static synchronized void ensureVertexIDConsistency(long l)
Method used to ensure consistency between internal atomic integer and vertex id from imported graphs.
static synchronized long getUniqueVertexIndex()
Unique counter for the number of graph vertices generated.
Tool to generate random numbers and random decisions.
boolean nextBoolean()
Returns the next pseudo-random, uniformly distributed boolean value from this random number generator...
public< T > T randomlyChooseOne(Collection< T > c)
Chooses one member among the given collection.
Enum specifying to what extent the template's inner graph can be changed.
FIXED
Inner graphs are effectively equivalent to the Fragment class, as no change in the inner structure is...
FIXED_STRUCT
Inner graph keep the same structure, but the identify of vertices can change.
The type of building block.
static BBType parseInt(int i)
Translates the integer into the enum.
FAILEDMUTATTEMTS_PERFORM_NOADDLINK_EDIT
FAILEDMUTATTEMTS_PERFORM_NODELETE
FAILEDMUTATTEMTS_PERFORM_NOADDRING
FAILEDMUTATTEMTS_PERFORM_NOADDLINK_FIND
FAILEDMUTATTEMTS_PERFORM_NODELLINK_EDIT
FAILEDMUTATTEMTS_PERFORM_NOADDRING_NORINGCOMB
FAILEDMUTATTEMTS_PERFORM_NOADDFUSEDRING_NOSITE
FAILEDMUTATTEMTS_PERFORM_NOADDFUSEDRING_NOBRIDGE
FAILEDMUTATTEMTS_PERFORM_NOOWNER
FAILEDMUTATTEMTS_PERFORM_NOADDLINK
FAILEDMUTATTEMTS_PERFORM_NOEXTEND
FAILEDMUTATTEMTS_PERFORM_NOMUTSITE
FAILEDMUTATTEMTS_PERFORM_NOADDFUSEDRING_NOFREEAP
FAILEDMUTATTEMTS_PERFORM_NODELETECHAIN
FAILEDMUTATTEMTS_PERFORM_NOCHANGELINK_EDIT
FAILEDMUTATTEMTS_PERFORM_NOCHANGELINK
FAILEDMUTATTEMTS_PERFORM_NOCHANGEBRANCH
FAILEDMUTATTEMTS_PERFORM_NOADDRING_NOFREEAP
FAILEDMUTATTEMTS_PERFORM_BADMUTTYPE
FAILEDMUTATTEMTS_PERFORM_NOCHANGELINK_FIND
Identifier of the type of parameters.
FS_PARAMS
Parameters pertaining the definition of the fragment space.
RC_PARAMS
Parameters pertaining to ring closures in graphs.
Types of crossover defined.
SUBGRAPH
Swaps a portion of a branch trying to retain cyclicity.
BRANCH
Swaps the entire branch starting from a given vertex.
Types of mutation defined in relation to what happens to the target vertex (i.e., the actual mutation...
DELETECHAIN
Removes a vertex and all its neighbors recursively until a branching point, i.e., until a vertex that...