001/* 002 * BioJava development code 003 * 004 * This code may be freely distributed and modified under the 005 * terms of the GNU Lesser General Public Licence. This should 006 * be distributed with the code. If you do not have a copy, 007 * see: 008 * 009 * http://www.gnu.org/copyleft/lesser.html 010 * 011 * Copyright for this code is held jointly by the individual 012 * authors. These should be listed in @author doc comments. 013 * 014 * For more information on the BioJava project and its aims, 015 * or to join the biojava-l mailing list, visit the home page 016 * at: 017 * 018 * http://www.biojava.org/ 019 * 020 */ 021package org.biojava.nbio.structure.symmetry.internal; 022 023import java.util.ArrayList; 024import java.util.Collections; 025import java.util.List; 026import java.util.Set; 027 028import javax.vecmath.GMatrix; 029import javax.vecmath.GVector; 030 031import org.biojava.nbio.structure.Atom; 032import org.biojava.nbio.structure.StructureException; 033import org.biojava.nbio.structure.align.model.AFPChain; 034import org.biojava.nbio.structure.align.multiple.MultipleAlignment; 035import org.biojava.nbio.structure.align.util.AlignmentTools; 036import org.biojava.nbio.structure.symmetry.utils.SymmetryTools; 037import org.jgrapht.Graph; 038import org.jgrapht.alg.connectivity.ConnectivityInspector; 039import org.jgrapht.graph.DefaultEdge; 040 041/** 042 * The GraphRefiner transforms the self-alignment into a Graph and extracts its 043 * maximally connected Components. It then refines the alignment by combining 044 * the compatible Components with the following heuristic: 045 * 046 * <pre> 047 * Given a set of components and their pairwise compatibilities, iteratively 048 * add the most compatible component, which is compatible to all the components 049 * already added, to the final alignment. 050 * </pre> 051 * 052 * @author Aleix Lafita 053 * 054 */ 055public class GraphComponentRefiner implements SymmetryRefiner { 056 057 @Override 058 public MultipleAlignment refine(AFPChain selfAlignment, Atom[] atoms, int order) 059 throws StructureException, RefinerFailedException { 060 061 // Construct the alignment graph with jgrapht 062 Graph<Integer, DefaultEdge> graph = SymmetryTools 063 .buildSymmetryGraph(selfAlignment); 064 065 // Find the maximally connected components of the graph 066 ConnectivityInspector<Integer, DefaultEdge> inspector = new ConnectivityInspector<Integer, DefaultEdge>( 067 graph); 068 List<Set<Integer>> components = inspector.connectedSets(); 069 070 // Filter components with size != order, and transform to ResidueGroups 071 List<ResidueGroup> groups = new ArrayList<>(); 072 for (Set<Integer> comp : components) { 073 if (comp.size() == order) { 074 ResidueGroup group = new ResidueGroup(comp); 075 groups.add(group); 076 } 077 } 078 int size = groups.size(); 079 if (size == 0) 080 throw new RefinerFailedException("Could not find any components " 081 + "in the alignment Graph of size != order."); 082 083 // Create a square matrix of component compatibility 084 GMatrix matrix = new GMatrix(size, size); 085 for (int i = 0; i < size; i++) { 086 for (int j = i; j < size; j++) { 087 // The diagonal is always 0 088 if (i == j){ 089 matrix.setElement(i, j, 0); 090 continue; 091 } 092 // If compatible put 1, otherwise 0 093 ResidueGroup g1 = groups.get(i); 094 ResidueGroup g2 = groups.get(j); 095 if (g1.isCompatible(g2)) { 096 matrix.setElement(i, j, 1); 097 matrix.setElement(j, i, 1); 098 } else { 099 matrix.setElement(i, j, 0); 100 matrix.setElement(j, i, 0); 101 } 102 } 103 } 104 105 // The compatibility score is the sum of rows of the matrix 106 List<Integer> rowScores = new ArrayList<>(size); 107 for (int i = 0; i < size; i++) { 108 GVector row = new GVector(size); 109 matrix.getRow(i, row); 110 // because element={0,1}, this is the sum 111 int rowScore = (int) row.normSquared(); 112 rowScores.add(rowScore); 113 } 114 115 // Refined multiple alignment Block as a result 116 List<List<Integer>> alignRes = new ArrayList<>(order); 117 for (int i = 0; i < order; i++) 118 alignRes.add(new ArrayList<Integer>()); 119 120 // Iterate until no more groups left to add (all groups score 0) 121 while (true) { 122 // Take the most compatible ResidueGroup 123 Integer max = Collections.max(rowScores); 124 int index = rowScores.indexOf(max); 125 126 // Add the group to the alignment Block 127 groups.get(index).combineWith(alignRes); 128 129 // Zero all the scores of incompatible groups 130 boolean allZero = true; 131 for (int i=0; i<size; i++){ 132 if (matrix.getElement(index, i) < 1.0) 133 rowScores.set(i, 0); 134 else if (rowScores.get(i) != 0) 135 allZero = false; 136 } 137 if (allZero) 138 break; 139 } 140 141 for (int i = 0; i < order; i++) 142 Collections.sort(alignRes.get(i)); 143 144 int length = alignRes.get(0).size(); 145 if (length == 0) 146 throw new RefinerFailedException("Empty alignment"); 147 148 int[][][] optAln = new int[order][2][length]; 149 for (int bk = 0; bk < order; bk++) { 150 optAln[bk] = new int[2][]; 151 optAln[bk][0] = new int[length]; 152 optAln[bk][1] = new int[length]; 153 for (int pos = 0; pos < length; pos++) { 154 optAln[bk][0][pos] = alignRes.get(bk).get(pos); 155 optAln[bk][1][pos] = alignRes.get((bk + 1) % order).get(pos); 156 } 157 } 158 AFPChain afp = AlignmentTools.replaceOptAln(optAln, selfAlignment, atoms, atoms); 159 return SymmetryTools.fromAFP(afp, atoms); 160 } 161 162}