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.Arrays;
025import java.util.List;
026
027import org.biojava.nbio.structure.Atom;
028import org.biojava.nbio.structure.PdbId;
029import org.biojava.nbio.structure.ResidueNumber;
030import org.biojava.nbio.structure.ResidueRange;
031import org.biojava.nbio.structure.StructureException;
032import org.biojava.nbio.structure.StructureIdentifier;
033import org.biojava.nbio.structure.SubstructureIdentifier;
034import org.biojava.nbio.structure.align.model.AFPChain;
035import org.biojava.nbio.structure.align.multiple.Block;
036import org.biojava.nbio.structure.align.multiple.MultipleAlignment;
037import org.biojava.nbio.structure.align.multiple.util.MultipleAlignmentScorer;
038import org.biojava.nbio.structure.symmetry.internal.CESymmParameters.RefineMethod;
039import org.biojava.nbio.structure.symmetry.internal.CESymmParameters.SymmetryType;
040import org.biojava.nbio.structure.symmetry.utils.SymmetryTools;
041
042/**
043 * This Class stores all the relevant information of an internal symmetry result
044 * obtained with CeSymm. The purpose is to carry all the information packed
045 * during the calculations and return a single object.
046 *
047 * @author Aleix Lafita
048 * @since 4.2.0
049 *
050 */
051public class CeSymmResult {
052
053        private MultipleAlignment multipleAlignment;
054        private AFPChain selfAlignment;
055
056        private StructureIdentifier structureId;
057        private Atom[] atoms;
058
059        private CESymmParameters params;
060        private SymmetryAxes axes = new SymmetryAxes();
061        private String symmGroup;
062
063        private int numRepeats;
064        private boolean refined;
065
066        /**
067         * Conditions checked are: score above the threshold, number of repeats
068         * higher than 1 and refinement succeeded.
069         *
070         * @return true if significant, false otherwise
071         */
072        public boolean isSignificant() {
073
074                // In any case if the order is 1 it is asymmetric
075                if (numRepeats < 2)
076                        return false;
077
078                // If the TM-Score before refinement is below the threshold
079                if (selfAlignment.getTMScore() < params.getUnrefinedScoreThreshold())
080                        return false;
081
082                // If the refinement was attempted
083                if (params.getRefineMethod() != RefineMethod.NOT_REFINED) {
084                        // Check for minimum length
085                        if (multipleAlignment == null || multipleAlignment.getCoreLength() < params.getMinCoreLength())
086                                return false;
087                        // Allow 90% of original TM-Score theshold
088                        if (multipleAlignment.getScore(MultipleAlignmentScorer.AVGTM_SCORE) < params
089                                        .getRefinedScoreThreshold())
090                                return false;
091                        return true;
092                }
093
094                return true;
095        }
096
097        /**
098         * Return the symmetric repeats as structure identifiers, if the result is
099         * symmetric and it was refined, return null otherwise.
100         *
101         * @return List of StructureIdentifiers or null if not defined
102         * @throws StructureException
103         */
104        public List<StructureIdentifier> getRepeatsID() throws StructureException {
105
106                if (!isRefined())
107                        return null;
108
109                List<StructureIdentifier> repeats = new ArrayList<>(
110                                numRepeats);
111
112                PdbId pdbId = structureId.toCanonical().getPdbId();
113                Block align = multipleAlignment.getBlocks().get(0);
114
115                for (int su = 0; su < numRepeats; su++) {
116                        // Get the start and end residues of the repeat
117                        ResidueNumber res1 = atoms[align.getStartResidue(su)].getGroup()
118                                        .getResidueNumber();
119                        ResidueNumber res2 = atoms[align.getFinalResidue(su)].getGroup()
120                                        .getResidueNumber();
121                        ResidueRange range = new ResidueRange(res1.getChainName(), res1, res2);
122
123                        StructureIdentifier id = new SubstructureIdentifier(pdbId,
124                                        Arrays.asList(range));
125
126                        repeats.add(id);
127                }
128                return repeats;
129        }
130
131        @Override
132        public String toString() {
133                return structureId + ", symmGroup=" + getSymmGroup() + ", numRepeats="
134                                + numRepeats + ", symmLevels=" + axes.getNumLevels()
135                                + ", refined=" + refined + " | " + params;
136        }
137
138        public MultipleAlignment getMultipleAlignment() {
139                return multipleAlignment;
140        }
141
142        public void setMultipleAlignment(MultipleAlignment multipleAlignment) {
143                this.multipleAlignment = multipleAlignment;
144        }
145
146        public AFPChain getSelfAlignment() {
147                return selfAlignment;
148        }
149
150        public void setSelfAlignment(AFPChain selfAlignment) {
151                this.selfAlignment = selfAlignment;
152        }
153
154        public CESymmParameters getParams() {
155                return params;
156        }
157
158        public void setParams(CESymmParameters params) {
159                this.params = params.clone();
160        }
161
162        public SymmetryAxes getAxes() {
163                return axes;
164        }
165
166        public void setAxes(SymmetryAxes axes) {
167                this.axes = axes;
168        }
169
170        /**
171         * Return the symmetry order determined by the order detector if the
172         * symmetry is significant. Return 1 otherwise.
173         *
174         * @return the order of symmetry if the result is significant
175         */
176        public int getNumRepeats() {
177                if (isSignificant())
178                        return numRepeats;
179                else
180                        return 1;
181        }
182
183        public void setNumRepeats(int symmOrder) {
184                this.numRepeats = symmOrder;
185        }
186
187        public boolean isRefined() {
188                return refined;
189        }
190
191        public void setRefined(boolean refined) {
192                this.refined = refined;
193        }
194
195        public String getSymmGroup() {
196                // Lazily calculate the symmetry group if significant
197                if (symmGroup == null) {
198                        if (isSignificant()) {
199                                if (isRefined()) {
200                                        try {
201                                                symmGroup = SymmetryTools.getQuaternarySymmetry(this)
202                                                                .getSymmetry();
203                                        } catch (StructureException e) {
204                                                symmGroup = "C1";
205                                        }
206                                        if ("C1".equals(symmGroup))
207                                                symmGroup = "R"; // could not find group
208                                } else {
209                                        // in case significant but not refined
210                                        if (axes.getElementaryAxis(0).getSymmType()
211                                                        .equals(SymmetryType.CLOSED))
212                                                symmGroup = "C" + numRepeats;
213                                        else
214                                                symmGroup = "R";
215                                }
216                        } else
217                                // case asymmetric
218                                symmGroup = "C1";
219                }
220                return symmGroup;
221        }
222
223        public void setSymmGroup(String symmGroup) {
224                this.symmGroup = symmGroup;
225        }
226
227        public Atom[] getAtoms() {
228                return atoms;
229        }
230
231        public void setAtoms(Atom[] atoms) {
232                this.atoms = atoms;
233        }
234
235        public int getSymmLevels() {
236                return axes.getNumLevels();
237        }
238
239        public StructureIdentifier getStructureId() {
240                return structureId;
241        }
242
243        public void setStructureId(StructureIdentifier structureId) {
244                this.structureId = structureId;
245        }
246
247        /**
248         * Return a String describing the reasons for the CE-Symm final decision in
249         * this particular result.
250         *
251         * @return String decision reason
252         */
253        public String getReason() {
254                // Cases:
255                // 1. Asymmetric because insignificant self-alignment (1itb.A_1-100)
256                double tm = selfAlignment.getTMScore();
257                if (tm < params.getUnrefinedScoreThreshold()) {
258                        return String.format("Insignificant self-alignment (TM=%.2f)", tm);
259                }
260                // 2. Asymmetric because order detector returned 1
261                if (numRepeats == 1) {
262                        return String.format(
263                                        "Order detector found asymmetric alignment (TM=%.2f)", tm);
264                }
265
266                // Check that the user requested refinement
267                if (params.getRefineMethod() != RefineMethod.NOT_REFINED) {
268                        // 3. Asymmetric because refinement failed
269                        if (!refined) {
270                                return "Refinement failed";
271                        }
272                        tm = multipleAlignment
273                                        .getScore(MultipleAlignmentScorer.AVGTM_SCORE);
274                        // 4. Asymmetric because refinement & optimization were not
275                        // significant
276                        if (!isSignificant()) {
277                                return String.format(
278                                                "Refinement was not significant (TM=%.2f)", tm);
279                        }
280                } else {
281                        // 4. Not refined, but result was not significant
282                        if (!isSignificant()) {
283                                return String
284                                                .format("Result was not significant (TM=%.2f)", tm);
285                        }
286                }
287
288                String hierarchical = "";
289                if (axes.getNumLevels() > 1) {
290                        hierarchical = String.format("; Contains %d levels of symmetry",
291                                        axes.getNumLevels());
292                }
293                // 5. Symmetric.
294                // a. Open. Give # repeats (1n0r.A)
295                if (axes.getElementaryAxis(0).getSymmType() == SymmetryType.OPEN) {
296                        return String.format("Contains %d open repeats (TM=%.2f)%s",
297                                        getNumRepeats(), tm, hierarchical);
298                }
299                // b. Closed, non-hierarchical (1itb.A)
300                // c. Closed, heirarchical (4gcr)
301                return String.format("Significant (TM=%.2f)%s", tm, hierarchical);
302        }
303
304}