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.align.seq; 022 023import org.biojava.nbio.alignment.Alignments; 024import org.biojava.nbio.alignment.Alignments.PairwiseSequenceAlignerType; 025import org.biojava.nbio.alignment.SimpleGapPenalty; 026import org.biojava.nbio.core.alignment.matrices.SubstitutionMatrixHelper; 027import org.biojava.nbio.alignment.template.GapPenalty; 028import org.biojava.nbio.alignment.template.PairwiseSequenceAligner; 029import org.biojava.nbio.core.alignment.template.SequencePair; 030import org.biojava.nbio.core.alignment.template.SubstitutionMatrix; 031import org.biojava.nbio.structure.Atom; 032import org.biojava.nbio.structure.Group; 033import org.biojava.nbio.structure.StructureException; 034import org.biojava.nbio.structure.StructureTools; 035import org.biojava.nbio.structure.align.AbstractStructureAlignment; 036import org.biojava.nbio.structure.align.StructureAlignment; 037import org.biojava.nbio.structure.align.ce.CECalculator; 038import org.biojava.nbio.structure.align.ce.CeParameters; 039import org.biojava.nbio.structure.align.ce.ConfigStrucAligParams; 040import org.biojava.nbio.structure.align.ce.UserArgumentProcessor; 041import org.biojava.nbio.structure.align.model.AFPChain; 042import org.biojava.nbio.structure.align.util.AFPAlignmentDisplay; 043import org.biojava.nbio.structure.align.util.AlignmentTools; 044import org.biojava.nbio.structure.align.util.ConfigurationException; 045import org.biojava.nbio.core.exceptions.CompoundNotFoundException; 046import org.biojava.nbio.core.sequence.ProteinSequence; 047import org.biojava.nbio.core.sequence.compound.AminoAcidCompound; 048import org.biojava.nbio.core.sequence.compound.AminoAcidCompoundSet; 049import org.biojava.nbio.core.sequence.template.Compound; 050import org.slf4j.Logger; 051import org.slf4j.LoggerFactory; 052 053/** 054 * Provides a 3D superimposition of two structures based on their sequence 055 * alignment. 056 * <p> 057 * This algorithm includes a final step to iteratively drop columns of the 058 * alignment until a maximum RMSD threshold of the superimposition, or the 059 * minimum alignment length theshold, are fulfilled, similar to what pymol align 060 * algorithm does. 061 * 062 * @author Andreas Prlic 063 * @author Aleix Lafita 064 * 065 */ 066public class SmithWaterman3Daligner extends AbstractStructureAlignment implements StructureAlignment { 067 068 public static final String algorithmName = "Smith-Waterman superposition"; 069 070 private static final Logger logger = LoggerFactory.getLogger(SmithWaterman3Daligner.class); 071 072 /** 073 * version history: 074 * 1.1 - Added a maxRMSD and minLen parameters 075 * 1.0 - Initial version 076 */ 077 private static final String version = "1.1"; 078 private SmithWaterman3DParameters params; 079 080 public static void main(String[] args) throws ConfigurationException { 081 //args = new String[]{"-pdb1","1cdg.A","-pdb2","1tim.A","-pdbFilePath","/tmp/","-show3d","-printFatCat"}; 082 UserArgumentProcessor processor = new SmithWatermanUserArgumentProcessor(); 083 processor.process(args); 084 } 085 086 public SmithWaterman3Daligner(){ 087 params = new SmithWaterman3DParameters(); 088 } 089 090 @Override 091 public AFPChain align(Atom[] ca1, Atom[] ca2) throws StructureException { 092 if ( params == null) 093 params = new SmithWaterman3DParameters(); 094 return align(ca1,ca2,params); 095 } 096 097 @Override 098 public AFPChain align(Atom[] ca1, Atom[] ca2, Object parameters) 099 throws StructureException { 100 101 if ( parameters == null) { 102 throw new IllegalArgumentException("Got null instead of SmithWaterman3DParameters!"); 103 } 104 if ( ! (parameters instanceof SmithWaterman3DParameters)) 105 throw new IllegalArgumentException("provided parameter object is not of type SmithWaterman3DParameters, but " + parameters.getClass().getName()); 106 107 params = (SmithWaterman3DParameters) parameters; 108 AFPChain afpChain = new AFPChain(algorithmName); 109 110 111 // covert input to sequences 112 String seq1 = StructureTools.convertAtomsToSeq(ca1); 113 String seq2 = StructureTools.convertAtomsToSeq(ca2); 114 115 ProteinSequence s1 = null; 116 ProteinSequence s2 = null; 117 118 try { 119 s1 = new ProteinSequence(seq1); 120 s2 = new ProteinSequence(seq2); 121 } catch (CompoundNotFoundException e){ 122 123 throw new StructureException(e.getMessage(),e); 124 } 125 126 // default blosum62 127 SubstitutionMatrix<AminoAcidCompound> matrix = SubstitutionMatrixHelper.getBlosum65(); 128 129 GapPenalty penalty = new SimpleGapPenalty(); 130 131 penalty.setOpenPenalty(params.getGapOpen()); 132 penalty.setExtensionPenalty(params.getGapExtend()); 133 134 PairwiseSequenceAligner<ProteinSequence, AminoAcidCompound> smithWaterman = 135 Alignments.getPairwiseAligner(s1, s2, PairwiseSequenceAlignerType.LOCAL, penalty, matrix); 136 137 SequencePair<ProteinSequence, AminoAcidCompound> pair = smithWaterman.getPair(); 138 139 if (pair.getTarget().toString().isEmpty() || pair.getQuery().toString().isEmpty()) { 140 throw new StructureException("Empty alignment for sequences "+s1+" and "+s2); 141 } 142 143 logger.debug("Smith-Waterman alignment is: {}", pair.toString(100)); 144 145 // convert to a 3D alignment... 146 afpChain = convert(ca1,ca2,pair, smithWaterman); 147 148 // Perform an iterative dropping of the columns 149 while (afpChain.getOptLength() > params.getMinLen() 150 && afpChain.getTotalRmsdOpt() > params.getMaxRmsd()) { 151 afpChain = AlignmentTools.deleteHighestDistanceColumn(afpChain, ca1, ca2); 152 } 153 154 return afpChain; 155 } 156 157 /** 158 * Converts a sequence alignment into a structural alignment 159 * @param smithWaterman The sequence aligner 160 * @param ca1 CA atoms from the query sequence 161 * @param ca2 CA atoms from the target sequence 162 * @param smithWaterman pairwise Sequence aligner 163 * @param pair The sequence alignment calculated by aligner 164 * @return an AFPChain encapsulating the alignment in aligPair 165 * @throws StructureException 166 */ 167 private AFPChain convert(Atom[] ca1, Atom[] ca2, SequencePair<ProteinSequence, 168 AminoAcidCompound> pair, PairwiseSequenceAligner<ProteinSequence, AminoAcidCompound> smithWaterman) throws StructureException { 169 AFPChain afpChain = new AFPChain(algorithmName); 170 int ca1Length = ca1.length; 171 int ca2Length = ca2.length; 172 173 afpChain.setAlignScore(smithWaterman.getScore()); 174 afpChain.setCa1Length(ca1Length); 175 afpChain.setCa2Length(ca2Length); 176 177 int nrCols = pair.getLength(); 178 int nAtom=0; 179 int nGaps=0; 180 181 Atom[] strBuf1 = new Atom[nrCols]; 182 Atom[] strBuf2 = new Atom[nrCols]; 183 char[] alnseq1 = new char[ca1Length+ca2Length+1]; 184 char[] alnseq2 = new char[ca1Length+ca2Length+1] ; 185 char[] alnsymb = new char[ca1Length+ca2Length+1]; 186 187 Compound gapSymbol = AminoAcidCompoundSet.getAminoAcidCompoundSet().getCompoundForString("-"); 188 189 int pos = 0 ; // aligned positions 190 191 int nrIdent = 0; 192 int nrSim = 0; 193 194 int[] align_se1 = new int[nrCols+1]; 195 int[] align_se2 = new int[nrCols+1]; 196 197 for ( int i = 1 ; i <= nrCols; i ++){ 198 199 int myI = i-1; 200 201 Compound s1 = pair.getCompoundAt(1, i); 202 Compound s2 = pair.getCompoundAt(2,i); 203 204 // alignment is using internal index start at 1... 205 int pos1 = pair.getIndexInQueryAt(i) -1; 206 int pos2 = pair.getIndexInTargetAt(i) -1; 207 208 if ( ( ! s1.equals(gapSymbol) )&& (! s2.equals(gapSymbol))){ 209 strBuf1[nAtom] = ca1[pos1]; 210 strBuf2[nAtom] = ca2[pos2]; 211 // 212 char l1 = getOneLetter(ca1[pos1].getGroup()); 213 char l2 = getOneLetter(ca2[pos2].getGroup()); 214 // 215 alnseq1[myI] = Character.toUpperCase(l1); 216 alnseq2[myI] = Character.toUpperCase(l2); 217 alnsymb[myI] = ' '; 218 // 219 if ( l1 == l2) { 220 nrIdent++; 221 nrSim++; 222 alnsymb[myI] = '|'; 223 224 } else if ( AFPAlignmentDisplay.aaScore(l1, l2) > 0){ 225 226 nrSim++; 227 alnsymb[myI] = ':'; 228 } 229 // 230 align_se1[myI] = pos1; 231 align_se2[myI] = pos2; 232 // 233 pos++; 234 nAtom++; 235 236 } else { 237 // there is a gap at this position 238 nGaps++; 239 240 alnsymb[myI] = ' '; 241 align_se1[myI] = -1; 242 align_se2[myI] = -1; 243 244 if ( s1.equals(gapSymbol)){ 245 alnseq1[myI] = '-'; 246 247 } else { 248 char l1 = getOneLetter(ca1[pos1].getGroup()); 249 alnseq1[myI] = Character.toUpperCase(l1); 250 align_se1[myI] = pos1; 251 } 252 if ( s2.equals(gapSymbol)){ 253 alnseq2[myI] = '-'; 254 255 } else { 256 char l2 = getOneLetter(ca2[pos2].getGroup()); 257 alnseq2[myI] = Character.toUpperCase(l2); 258 align_se2[myI] = pos2; 259 260 } 261 } 262 263 } 264 265 afpChain.setAlnbeg1(pair.getIndexInQueryAt(1)-1); 266 afpChain.setAlnbeg2(pair.getIndexInTargetAt(1)-1); 267 268 afpChain.setGapLen(nGaps); 269 afpChain.setAlnseq1(alnseq1); 270 271 afpChain.setAlnseq2(alnseq2); 272 afpChain.setAlnsymb(alnsymb); 273 274 // CE uses the aligned pairs as reference not the whole alignment including gaps... 275 afpChain.setIdentity(nrIdent*1.0/pos); 276 afpChain.setSimilarity(nrSim*1.0/pos); 277 278 afpChain.setAlnLength(nrCols); 279 afpChain.setOptLength(nAtom); 280 int[] optLen = new int[]{nAtom}; 281 afpChain.setOptLen(optLen); 282 283 if(nAtom<4) 284 return afpChain; 285 286 CeParameters params = new CeParameters(); 287 CECalculator cecalc = new CECalculator(params); 288 289 // here we don't store the rotation matrix for the user! 290 291 double rmsd= cecalc.calc_rmsd(strBuf1, strBuf2, nAtom,true); 292 293 afpChain.setBlockRmsd(new double[]{rmsd}); 294 afpChain.setOptRmsd(new double[]{rmsd}); 295 afpChain.setTotalRmsdOpt(rmsd); 296 afpChain.setChainRmsd(rmsd); 297 298 // let's hijack the CE implementation 299 // and use some utilities from there to 300 // build up the afpChain object 301 302 cecalc.setnAtom(nAtom); 303 304 cecalc.setAlign_se1(align_se1); 305 cecalc.setAlign_se2(align_se2); 306 307 cecalc.setLcmp(nrCols ); 308 309 cecalc.convertAfpChain(afpChain, ca1, ca2); 310 311 afpChain.setAlgorithmName(algorithmName); 312 afpChain.setVersion(version); 313 314 return afpChain; 315 } 316 317 private static char getOneLetter(Group g){ 318 319 if (g==null) return StructureTools.UNKNOWN_GROUP_LABEL; 320 321 return StructureTools.get1LetterCode(g.getPDBName()); 322 323 } 324 325 @Override 326 public String getAlgorithmName() { 327 return algorithmName; 328 } 329 330 @Override 331 public ConfigStrucAligParams getParameters() { 332 return params; 333 } 334 335 @Override 336 public String getVersion() { 337 return version; 338 } 339 340 @Override 341 public void setParameters(ConfigStrucAligParams parameters) { 342 if ( ! (parameters instanceof SmithWaterman3DParameters)) 343 throw new IllegalArgumentException("provided parameter object is not of type SmithWaterman3DParameters"); 344 params = (SmithWaterman3DParameters)parameters; 345 } 346 347}