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.secstruc;
022
023import java.io.BufferedReader;
024import java.io.File;
025import java.io.FileReader;
026import java.io.IOException;
027import java.io.InputStream;
028import java.io.InputStreamReader;
029import java.io.Reader;
030import java.io.StringReader;
031import java.util.ArrayList;
032import java.util.List;
033
034import org.biojava.nbio.structure.Group;
035import org.biojava.nbio.structure.ResidueNumber;
036import org.biojava.nbio.structure.Structure;
037import org.biojava.nbio.structure.StructureException;
038import org.slf4j.Logger;
039import org.slf4j.LoggerFactory;
040
041/**
042 * Class to parse a DSSP file (output of the DSSP program),
043 * that contains the secondary structure assignment of a structure.
044 * <p>
045 * This class has been ported from the OWL Java library for
046 * Structural Bioinformatics (https://github.com/eppic-team/owl).
047 * <p>
048 * As of September 2015, the DSSP source code and executables can
049 * be downloaded from http://swift.cmbi.ru.nl/gv/dssp/.
050 *
051 * @author Aleix Lafita
052 * @since 4.1.1
053 *
054 */
055public class DSSPParser {
056
057        private static final Logger logger =
058                        LoggerFactory.getLogger(DSSPParser.class);
059
060        /**
061         * Parse a DSSP output file and return the secondary structure
062         * annotation as a List of {@link SecStrucState} objects.
063         *
064         * @param dsspIs an InputStream to a DSSP file
065         * @param structure Structure object associated to the dssp
066         * @param assign assigns the SS to the structure if true
067         * @return a List of SS annotation objects
068         * @throws StructureException
069         * @throws IOException
070         */
071        public static List<SecStrucState> parseInputStream(InputStream dsspIs,
072                        Structure structure, boolean assign)
073                                        throws IOException, StructureException {
074
075                BufferedReader reader = new BufferedReader(new InputStreamReader(dsspIs));
076                return generalParse(reader, structure, assign);
077        }
078
079        /**
080         * Parse a DSSP output file and return the secondary structure
081         * annotation as a List of {@link SecStrucState} objects.
082         *
083         * @param dsspPath path to the DSSP file to parse
084         * @param structure Structure object associated to the dssp
085         * @param assign assigns the SS to the structure if true
086         * @return a List of SS annotation objects
087         * @throws StructureException
088         * @throws IOException
089         */
090        public static List<SecStrucState> parseFile(String dsspPath,
091                        Structure structure, boolean assign)
092                                        throws IOException, StructureException {
093
094                File file = new File(dsspPath);
095                Reader read = new FileReader(file);
096                BufferedReader reader = new BufferedReader(read);
097                return generalParse(reader, structure, assign);
098        }
099
100        /**
101         * Parse a DSSP format String and return the secondary structure
102         * annotation as a List of {@link SecStrucState} objects.
103         *
104         * @param dsspOut String with the DSSP output to parse
105         * @param structure Structure object associated to the dssp
106         * @param assign assigns the SS to the structure if true
107         * @return a List of SS annotation objects
108         * @throws StructureException
109         * @throws IOException
110         */
111        public static List<SecStrucState> parseString(String dsspOut,
112                        Structure structure, boolean assign)
113                                        throws IOException, StructureException {
114
115                Reader read = new StringReader(dsspOut);
116                BufferedReader reader = new BufferedReader(read);
117                return generalParse(reader, structure, assign);
118        }
119
120        private static List<SecStrucState> generalParse(BufferedReader reader,
121                        Structure structure, boolean assign)
122                                        throws IOException, StructureException {
123
124                String startLine = "  #  RESIDUE AA STRUCTURE BP1 BP2  ACC";
125                String line;
126
127                List<SecStrucState> secstruc = new ArrayList<>();
128
129                //Find the first line of the DSSP output
130                while((line = reader.readLine()) != null) {
131                        if(line.startsWith(startLine)) break;
132                }
133
134                while((line = reader.readLine()) != null) {
135
136                        String indexStr = line.substring(0,5).trim();
137                        String resNumStr = line.substring(5,10).trim();
138
139                        //Only happens if dssp inserts a line indicating a chain break
140                        if(!"".equals(resNumStr)) {
141
142                                int index = Integer.parseInt(indexStr);
143                                //Get the group of the structure corresponding to the residue
144                                int resNum = Integer.parseInt(resNumStr);
145                                char insCode = line.charAt(10);
146                                String chainId = line.substring(11,13).trim();
147                                ResidueNumber r = new ResidueNumber(chainId, resNum, insCode);
148                                Group parent = structure.getPolyChainByPDB(chainId)
149                                                .getGroupByPDB(r);
150                                SecStrucType ssType =
151                                                SecStrucType.fromCharacter(line.charAt(16));
152
153                                SecStrucState ss = new SecStrucState(parent,
154                                                SecStrucInfo.DSSP_ASSIGNMENT, ssType);
155
156                                //Parse the Bridge partners - TODO parallel or antiparallel?
157                                String bp = line.substring(25,29).trim();
158                                if (!"".equals(bp)) {
159                                        BetaBridge bb = new BetaBridge(
160                                                        index, Integer.parseInt(bp), BridgeType.parallel);
161                                        ss.addBridge(bb);
162                                } else logger.warn("Unable to parse beta Bridge for resn "+index);
163
164                                bp = line.substring(29,33).trim();
165                                if (!"".equals(bp)) {
166                                        BetaBridge bb = new BetaBridge(
167                                                        index, Integer.parseInt(bp), BridgeType.parallel);
168                                        ss.addBridge(bb);
169                                } else logger.warn("Unable to parse beta Bridge for resn "+index);
170
171                                //Parse the energy terms of donor and acceptor
172                                for (int i=0; i<4; i++){
173
174                                        int a = 42 + i*11;
175                                        int b = a + 8;
176
177                                        String val = line.substring(a,b).trim();
178                                        if ("".equals(val)) {
179                                                logger.warn("Unable to parse energy for resn "+index);
180                                                continue;
181                                        }
182
183                                        String[] p = val.split(",");
184
185                                        int partner = Integer.parseInt(p[0]);
186                                        if (partner != 0) partner += index;
187                                        double energy = Double.parseDouble(p[1]) * 1000.0;
188
189                                        switch(i){
190                                        case 0:
191                                                ss.getAccept1().setPartner(partner);
192                                                ss.getAccept1().setEnergy(energy);
193                                                break;
194                                        case 1:
195                                                ss.getDonor1().setPartner(partner);
196                                                ss.getDonor1().setEnergy(energy);
197                                                break;
198                                        case 2:
199                                                ss.getAccept2().setPartner(partner);
200                                                ss.getAccept2().setEnergy(energy);
201                                                break;
202                                        case 3:
203                                                ss.getDonor2().setPartner(partner);
204                                                ss.getDonor1().setEnergy(energy);
205                                                break;
206                                        }
207                                }
208
209                                //Angle properties
210                                String val = line.substring(91,97).trim();
211                                if (!"".equals(val)) ss.setKappa(Float.parseFloat(val));
212                                else logger.warn("Unable to parse kappa for resn "+index);
213
214                                val = line.substring(103,109).trim();
215                                if (!"".equals(val)) ss.setPhi(Float.parseFloat(val));
216                                else logger.warn("Unable to parse phi for resn "+index);
217
218                                val = line.substring(109,116).trim();
219                                if (!"".equals(val)) ss.setPsi(Float.parseFloat(val));
220                                else logger.warn("Unable to parse psi for resn "+index);
221
222                                if (assign) parent.setProperty(Group.SEC_STRUC, ss);
223                                secstruc.add(ss);
224                        }
225                }
226
227                reader.close();
228                return secstruc;
229        }
230
231}