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; 022 023 024import org.biojava.nbio.structure.Atom; 025import org.biojava.nbio.structure.Structure; 026import org.biojava.nbio.structure.StructureTools; 027import org.biojava.nbio.structure.align.ce.ConfigStrucAligParams; 028import org.biojava.nbio.structure.align.client.PdbPair; 029import org.biojava.nbio.structure.align.model.AFPChain; 030import org.biojava.nbio.structure.align.util.AtomCache; 031import org.biojava.nbio.structure.align.util.SynchronizedOutFile; 032import org.biojava.nbio.structure.align.xml.AFPChainXMLConverter; 033import org.slf4j.Logger; 034import org.slf4j.LoggerFactory; 035 036import java.io.File; 037import java.io.FileOutputStream; 038import java.io.OutputStreamWriter; 039import java.util.concurrent.Callable; 040import java.util.zip.GZIPOutputStream; 041 042/** 043 * Simple Callable Class that calculates a pairwise alignment in a different 044 * thread, so that multiple pairwise alignments can be run in parallel 045 * (examples: all-to-all alignments, DB search alignments). 046 * Adapted to a more general implementation since 4.1.0, because before it 047 * was thought for DB search only. 048 * 049 * @author Aleix Lafita 050 */ 051public class CallableStructureAlignment implements Callable<AFPChain> { 052 053 private final static Logger logger = LoggerFactory.getLogger( 054 CallableStructureAlignment.class); 055 056 //Structure information 057 private PdbPair pair; 058 private AtomCache cache; 059 private Atom[] ca1; 060 private Atom[] ca2; 061 062 //File output information - for DB searches 063 private SynchronizedOutFile outFile; 064 private File outFileDir; 065 066 //Algorithm information 067 private String algorithmName; 068 private ConfigStrucAligParams params; 069 070 /** 071 * Default constructor. Used in DB search. 072 * Instantiates an empty object, everything has to be set independently. 073 */ 074 public CallableStructureAlignment() {} 075 076 /** 077 * Constructor for all-to-all alignment calculation. 078 * Used for MultipleMC seed alignment calculation, for example. 079 * 080 * @param ca1 Atoms to align of the first structure 081 * @param ca2 Atoms to align of the second structure 082 * @param algorithmName the pairwise aligner algorithm to use, a new 083 * instance will be created for each thread. 084 * @param params parameter bean for the alignment. 085 */ 086 public CallableStructureAlignment(Atom[] ca1, Atom[] ca2, String algorithmName, ConfigStrucAligParams params){ 087 this.ca1 = ca1; 088 this.ca2 = ca2; 089 this.algorithmName = algorithmName; 090 this.params = params; 091 } 092 093 @Override 094 public AFPChain call() throws Exception { 095 096 //Prepare the alignment algorithm 097 StructureAlignment algorithm = StructureAlignmentFactory.getAlgorithm(algorithmName); 098 if (params!=null) algorithm.setParameters(params); 099 100 AFPChain afpChain = null; 101 try { 102 //Download the Atoms if they are not provided from the outisde (DB searches usually) 103 if (ca1 == null) { 104 Structure structure1 = cache.getStructure(pair.getName1()); 105 ca1 = StructureTools.getRepresentativeAtomArray(structure1); 106 } else ca1 = StructureTools.cloneAtomArray(ca1); 107 108 Structure structure2 = null; 109 if (ca2 == null) { 110 structure2 = cache.getStructure(pair.getName2()); 111 ca2 = StructureTools.getRepresentativeAtomArray(structure2); 112 } else ca2 = StructureTools.cloneAtomArray(ca2); 113 114 afpChain = algorithm.align(ca1, ca2); 115 116 if (pair!=null){ 117 afpChain.setName1(pair.getName1()); 118 afpChain.setName2(pair.getName2()); 119 } 120 121 //Do not output anything if there is no File information 122 if (outFile != null && outFileDir != null){ 123 String desc2 = structure2.getPDBHeader().getDescription(); 124 if ( desc2 == null) 125 desc2=""; 126 afpChain.setDescription2(desc2); 127 String result = afpChain.toDBSearchResult(); 128 logger.info("{}", result); 129 130 outFile.write(result); 131 132 String xml = AFPChainXMLConverter.toXML(afpChain, ca1, ca2); 133 writeXML(outFileDir,pair.getName1(), pair.getName2(), xml); 134 } 135 136 } catch ( Exception e){ 137 logger.error("Exception: ", e); 138 } 139 return afpChain; 140 } 141 142 public PdbPair getPair() { 143 return pair; 144 } 145 146 public void setPair(PdbPair pair) { 147 this.pair = pair; 148 } 149 150 public AtomCache getCache() { 151 return cache; 152 } 153 154 public void setCache(AtomCache cache) { 155 this.cache = cache; 156 } 157 158 public SynchronizedOutFile getOutFile() { 159 return outFile; 160 } 161 162 public void setOutFile(SynchronizedOutFile outFile) { 163 this.outFile = outFile; 164 } 165 166 public Atom[] getCa1() { 167 return ca1; 168 } 169 170 public void setCa1(Atom[] ca1) { 171 this.ca1 = ca1; 172 } 173 174 private void writeXML(File outFileF, String name1, String name2, String xml) 175 { 176 try{ 177 // Create file 178 File newF = new File(outFileF, "dbsearch_" +name1+"_" + name2+".xml.gz"); 179 180 try (OutputStreamWriter writer = new OutputStreamWriter(new GZIPOutputStream(new FileOutputStream(newF)))) { 181 writer.write(xml); 182 } 183 } catch (Exception e){//Catch exception if any 184 logger.error("Exception: ", e); 185 } 186 } 187 188 public void setOutputDir(File outFileF) { 189 this.outFileDir = outFileF; 190 } 191 192 public void setAlgorithmName(String algorithmName) { 193 this.algorithmName = algorithmName; 194 } 195 196 public void setParameters(ConfigStrucAligParams parameters) { 197 this.params = parameters; 198 } 199}