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 * Created on Sep 15, 2009 021 * Author: Andreas Prlic 022 * 023 */ 024 025package org.biojava.nbio.structure.align.ce; 026 027 028import org.biojava.nbio.structure.Atom; 029import org.biojava.nbio.structure.Group; 030import org.biojava.nbio.structure.StructureException; 031import org.biojava.nbio.structure.align.AbstractStructureAlignment; 032import org.biojava.nbio.structure.align.StructureAlignment; 033import org.biojava.nbio.structure.align.model.AFPChain; 034import org.biojava.nbio.structure.jama.Matrix; 035 036/** 037 * The main class of the Java implementation of the Combinatorial Extension Algorithm (CE), 038 * as has been originally developed by I. Shindyalov and P.Bourne (1998). 039 * The original CE paper is available from here: <a href="http://peds.oxfordjournals.org/cgi/content/short/11/9/739">http://peds.oxfordjournals.org/cgi/content/short/11/9/739</a> 040 * 041 * For a demo of how to use this algorithm, visit the BioJava web site: 042 * <a href="https://github.com/biojava/biojava-tutorial/blob/master/structure/alignment.md">CE usage example</a>. 043 * 044 * The BioJava CE version is based on CE version 2.3 (2003 or 2004). 045 * 046 * @author Andreas Prlic. 047 * 048 */ 049public class CeMain extends AbstractStructureAlignment implements StructureAlignment { 050 051 public static final String algorithmName = "jCE"; 052 053 /** 054 * version history: 055 * 1.2 - Added more parameters to the command line, including -maxOptRMSD 056 * 1.1 - Additional parameters 057 * 1.0 - Initial port from C code 058 */ 059 public static final String version = "1.2"; 060 061 protected CeParameters params; 062 protected CECalculator calculator; 063 private Atom[] ca2clone; 064 065 public CeMain(){ 066 super(); 067 params = new CeParameters(); 068 calculator = new CECalculator(params); 069 } 070 071 072 /** 073 * Example Parameters: 074 * 075 * -pdbFilePath /tmp -autoFetch -printCE -pdb1 1cnv -pdb2 3cna 076 * 077 */ 078 public static void main(String[] args) throws Exception { 079 CeUserArgumentProcessor processor = new CeUserArgumentProcessor(); //Responsible for creating a CeMain instance 080 processor.process(args); 081 } 082 083 /** 084 * Align ca2 onto ca1. 085 */ 086 @Override 087 public AFPChain align(Atom[] ca1, Atom[] ca2, Object param) throws StructureException{ 088 if ( ! (param instanceof CeParameters)) 089 throw new IllegalArgumentException("CE algorithm needs an object of call CeParameters as argument."); 090 091 params = (CeParameters) param; 092 093 // we don't want to rotate input atoms, do we? 094 ca2clone = new Atom[ca2.length]; 095 096 int pos = 0; 097 for (Atom a : ca2){ 098 Group g = (Group)a.getGroup().clone(); // works because each group has only a CA atom 099 100 ca2clone[pos] = g.getAtom(a.getName()); 101 102 pos++; 103 } 104 105 calculator = new CECalculator(params); 106 107 //Build alignment ca1 to ca2-ca2 108 AFPChain afpChain = new AFPChain(algorithmName); 109 afpChain = calculator.extractFragments(afpChain, ca1, ca2clone); 110 calculator.traceFragmentMatrix( afpChain,ca1, ca2clone); 111 calculator.nextStep( afpChain,ca1, ca2clone); 112 113 afpChain.setAlgorithmName(getAlgorithmName()); 114 afpChain.setVersion(version); 115 116 // Try to guess names 117 118 if (ca1.length!=0 && ca1[0].getGroup().getChain()!=null && ca1[0].getGroup().getChain().getStructure()!=null) 119 afpChain.setName1(ca1[0].getGroup().getChain().getStructure().getName()); 120 121 if (ca2.length!=0 && ca2[0].getGroup().getChain()!=null && ca2[0].getGroup().getChain().getStructure()!=null) 122 afpChain.setName2(ca2[0].getGroup().getChain().getStructure().getName()); 123 124 if ( afpChain.getNrEQR() == 0) 125 return afpChain; 126 127 // Set the distance matrix 128 129 int winSize = params.getWinSize(); 130 int winSizeComb1 = (winSize-1)*(winSize-2)/2; 131 double[][] m = calculator.initSumOfDistances(ca1.length, ca2.length, winSize, winSizeComb1, ca1, ca2clone); 132 afpChain.setDistanceMatrix(new Matrix(m)); 133 afpChain.setSequentialAlignment(true); 134 135 return afpChain; 136 } 137 138 139 140 141 @Override 142 public AFPChain align(Atom[] ca1, Atom[] ca2) throws StructureException { 143 144 if (params == null) 145 params = new CeParameters(); 146 147 return align(ca1,ca2,params); 148 } 149 150 @Override 151 public String getAlgorithmName() { 152 153 return CeMain.algorithmName; 154 } 155 156 @Override 157 public ConfigStrucAligParams getParameters() { 158 159 return params; 160 } 161 162 @Override 163 public void setParameters(ConfigStrucAligParams params){ 164 if (! (params instanceof CeParameters )){ 165 throw new IllegalArgumentException("provided parameter object is not of type CeParameter"); 166 } 167 this.params = (CeParameters) params; 168 } 169 170 @Override 171 public String getVersion() { 172 return CeMain.version; 173 } 174 175 public CECalculator getCECalculator() { 176 return calculator; 177 } 178}