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 Mar 9, 2010
021 * Author: Spencer Bliven
022 *
023 */
024
025package org.biojava.nbio.structure.align.ce;
026
027
028import org.biojava.nbio.structure.Atom;
029import org.biojava.nbio.structure.StructureException;
030import org.biojava.nbio.structure.StructureTools;
031import org.biojava.nbio.structure.align.model.AFPChain;
032import org.biojava.nbio.structure.align.util.AFPChainScorer;
033import org.biojava.nbio.structure.align.util.AtomCache;
034import org.biojava.nbio.structure.jama.Matrix;
035
036import java.lang.reflect.InvocationTargetException;
037import java.util.ArrayList;
038import java.util.Collections;
039import java.util.List;
040import java.util.Scanner;
041
042/**
043 * A wrapper for {@link CeMain} which sets default parameters to be appropriate for finding
044 * circular permutations.
045 * <p>
046 * A circular permutation consists of a single cleavage point and rearrangement
047 * between two structures, for example:
048 * <pre>
049 * ABCDEFG
050 * DEFGABC
051 * </pre>
052 * @author Spencer Bliven.
053 *
054 */
055public class OptimalCECPMain extends CeMain {
056        private static final boolean debug = true;
057
058
059        public static final String algorithmName = "jCE Optimal Circular Permutation";
060
061        /**
062         *  version history:
063         *  1.0 - Initial version
064         */
065        public static final String version = "1.0";
066
067        protected OptimalCECPParameters params;
068
069        /**
070         *
071         */
072        public OptimalCECPMain() {
073                super();
074                params = new OptimalCECPParameters();
075        }
076
077        @Override
078        public String getAlgorithmName() {
079                return OptimalCECPMain.algorithmName;
080        }
081
082        @Override
083        public String getVersion() {
084                return OptimalCECPMain.version;
085        }
086
087        /**
088         * @return an {@link OptimalCECPParameters} object
089         */
090        @Override
091        public ConfigStrucAligParams getParameters() {
092                return params;
093        }
094
095        /**
096         * @param params Should be an {@link OptimalCECPParameters} object specifying alignment options
097         */
098        @Override
099        public void setParameters(ConfigStrucAligParams params){
100                if (! (params instanceof OptimalCECPParameters )){
101                        throw new IllegalArgumentException("provided parameter object is not of type CeParameter");
102                }
103                this.params = (OptimalCECPParameters) params;
104        }
105
106        /**
107         * Circularly permutes arr in place.
108         *
109         * <p>Similar to {@link Collections#rotate(List, int)} but with reversed
110         * direction. Perhaps it would be more efficient to use the Collections version?
111         * @param <T>
112         * @param arr The array to be permuted
113         * @param cp The number of residues to shift leftward, or equivalently, the index of
114         *  the first element after the permutation point.
115         */
116        private static <T> void permuteArray(T[] arr, int cp) {
117                // Allow negative cp points for convenience.
118                if(cp == 0) {
119                        return;
120                }
121                if(cp < 0) {
122                        cp = arr.length+cp;
123                }
124                if(cp < 0 || cp >= arr.length) {
125                        throw new ArrayIndexOutOfBoundsException(
126                                        "Permutation point ("+cp+") must be between -ca2.length and ca2.length-1" );
127                }
128
129                List<T> temp = new ArrayList<>(cp);
130
131                // shift residues left
132                for(int i=0;i<cp;i++) {
133                        temp.add(arr[i]);
134                }
135                for(int j=cp;j<arr.length;j++) {
136                        arr[j-cp]=arr[j];
137                }
138                for(int i=0;i<cp;i++) {
139                        arr[arr.length-cp+i] = temp.get(i);
140                }
141        }
142
143        /**
144         * Circularly permutes arr in place.
145         *
146         * <p>Similar to {@link Collections#rotate(List, int)} but with reversed
147         * direction. Perhaps it would be more efficient to use the Collections version?
148         * @param <T>
149         * @param arr The array to be permuted
150         * @param cp The number of residues to shift leftward, or equivalently, the index of
151         *  the first element after the permutation point.
152         *
153        private static void permuteArray(int[] arr, int cp) {
154                // Allow negative cp points for convenience.
155                if(cp == 0) {
156                        return;
157                }
158                if(cp < 0) {
159                        cp = arr.length+cp;
160                }
161                if(cp < 0 || cp >= arr.length) {
162                        throw new ArrayIndexOutOfBoundsException(
163                                        "Permutation point ("+cp+") must be between -ca2.length and ca2.length-1" );
164                }
165
166                List<Integer> temp = new ArrayList<Integer>(cp);
167
168                // shift residues left
169                for(int i=0;i<cp;i++) {
170                        temp.add(arr[i]);
171                }
172                for(int j=cp;j<arr.length;j++) {
173                        arr[j-cp]=arr[j];
174                }
175                for(int i=0;i<cp;i++) {
176                        arr[arr.length-cp+i] = temp.get(i);
177                }
178        }
179        */
180
181        /**
182         * Aligns ca1 with ca2 permuted by <i>cp</i> residues.
183         * <p><strong>WARNING:</strong> Modifies ca2 during the permutation. Be sure
184         * to make a copy before calling this method.
185         *
186         * @param ca1
187         * @param ca2
188         * @param param
189         * @param cp
190         * @return
191         * @throws StructureException
192         */
193        public AFPChain alignPermuted(Atom[] ca1, Atom[] ca2, Object param, int cp) throws StructureException {
194                // initial permutation
195                permuteArray(ca2,cp);
196
197                // perform alignment
198                AFPChain afpChain = super.align(ca1, ca2, param);
199
200                // un-permute alignment
201                permuteAFPChain(afpChain, -cp);
202
203                if(afpChain.getName2() != null) {
204                        afpChain.setName2(afpChain.getName2()+" CP="+cp);
205                }
206
207                // Specify the permuted
208                return afpChain;
209        }
210
211        /**
212         * Permute the second protein of afpChain by the specified number of residues.
213         * @param afpChain Input alignment
214         * @param cp Amount leftwards (or rightward, if negative) to shift the
215         * @return A new alignment equivalent to afpChain after the permutations
216         */
217        private static void permuteAFPChain(AFPChain afpChain, int cp) {
218
219                int ca2len = afpChain.getCa2Length();
220
221                //fix up cp to be positive
222                if(cp == 0) {
223                        return;
224                }
225                if(cp < 0) {
226                        cp = ca2len+cp;
227                }
228                if(cp < 0 || cp >= ca2len) {
229                        throw new ArrayIndexOutOfBoundsException(
230                                        "Permutation point ("+cp+") must be between -ca2.length and ca2.length-1" );
231                }
232
233                // Fix up optAln
234                permuteOptAln(afpChain,cp);
235
236                if(afpChain.getBlockNum() > 1)
237                        afpChain.setSequentialAlignment(false);
238                // fix up matrices
239                // ca1 corresponds to row indices, while ca2 corresponds to column indices.
240                afpChain.setDistanceMatrix(permuteMatrix(afpChain.getDistanceMatrix(),0,-cp));
241                // this is square, so permute both
242                afpChain.setDisTable2(permuteMatrix(afpChain.getDisTable2(),-cp,-cp));
243
244                //TODO fix up other AFP parameters?
245
246        }
247
248        /**
249         * Permutes <i>mat</i> by moving the rows of the matrix upwards by <i>cp</i>
250         * rows.
251         * @param mat The original matrix
252         * @param cpRows Number of rows upward to move entries
253         * @param cpCols Number of columns leftward to move entries
254         * @return The permuted matrix
255         */
256        private static Matrix permuteMatrix(Matrix mat, int cpRows, int cpCols) {
257                //fix up cp to be positive
258                if(cpRows == 0 && cpCols == 0) {
259                        return mat.copy();
260                }
261                if(cpRows < 0) {
262                        cpRows = mat.getRowDimension()+cpRows;
263                }
264                if(cpRows < 0 || cpRows >= mat.getRowDimension()) {
265                        throw new ArrayIndexOutOfBoundsException( String.format(
266                                        "Can't permute rows by %d: only %d rows.",
267                                        cpRows, mat.getRowDimension() )
268                        );
269                }
270
271                if(cpCols < 0) {
272                        cpCols = mat.getColumnDimension()+cpCols;
273                }
274                if(cpCols < 0 || cpCols >= mat.getColumnDimension()) {
275                        throw new ArrayIndexOutOfBoundsException( String.format(
276                                        "Can't permute cols by %d: only %d rows.",
277                                        cpCols, mat.getColumnDimension() )
278                        );
279                }
280
281                int[] rows = new int[mat.getRowDimension()];
282                for(int i=0;i<rows.length;i++) {
283                        rows[i] = (i+cpRows)%rows.length;
284                }
285                int[] cols = new int[mat.getColumnDimension()];
286                for(int i=0;i<cols.length;i++) {
287                        cols[i] = (i+cpCols)%cols.length;
288                }
289
290                Matrix newMat = mat.getMatrix(rows, cols);
291                assert(newMat.getRowDimension() == mat.getRowDimension());
292                assert(newMat.getColumnDimension() == mat.getColumnDimension());
293                assert(newMat.get(0, 0) ==
294                        mat.get(cpRows%mat.getRowDimension(), cpCols%mat.getColumnDimension()));
295
296
297                return newMat;
298        }
299
300        /**
301         * Modifies the {@link AFPChain#setOptAln(int[][][]) optAln} of an AFPChain
302         * by permuting the second protein.
303         * <p>
304         * Sets residue numbers in the second protein to <i>(i-cp)%len</i>
305         *
306         * @param afpChain
307         * @param cp Amount leftwards (or rightward, if negative) to shift the
308         */
309        private static void permuteOptAln(AFPChain afpChain, int cp)
310        {
311                int ca2len = afpChain.getCa2Length();
312
313                if( ca2len <= 0) {
314                        throw new IllegalArgumentException("No Ca2Length specified in "+afpChain);
315                }
316
317                // Allow negative cp points for convenience.
318                if(cp == 0) {
319                        return;
320                }
321                if(cp <= -ca2len || cp >= ca2len) {
322                        // could just take cp%ca2len, but probably its a bug if abs(cp)>=ca2len
323                        throw new ArrayIndexOutOfBoundsException( String.format(
324                                        "Permutation point %d must be between %d and %d for %s",
325                                        cp, 1-ca2len,ca2len-1, afpChain.getName2() ) );
326                }
327                if(cp < 0) {
328                        cp = cp + ca2len;
329                }
330
331                // the unprocessed alignment
332                int[][][] optAln = afpChain.getOptAln();
333                int[] optLen = afpChain.getOptLen();
334
335                // the processed alignment
336                List<List<List<Integer>>> blocks = new ArrayList<>(afpChain.getBlockNum()*2);
337
338                //Update residue indices
339                // newi = (oldi-cp) % N
340                for(int block = 0; block < afpChain.getBlockNum(); block++) {
341                        if(optLen[block]<1)
342                                continue;
343
344                        // set up storage for the current block
345                        List<List<Integer>> currBlock = new ArrayList<>(2);
346                        currBlock.add( new ArrayList<Integer>());
347                        currBlock.add( new ArrayList<Integer>());
348                        blocks.add(currBlock);
349
350                        // pos = 0 case
351                        currBlock.get(0).add( optAln[block][0][0] );
352                        currBlock.get(1).add( (optAln[block][1][0]+cp ) % ca2len);
353
354                        for(int pos = 1; pos < optLen[block]; pos++) {
355                                //check if we need to start a new block
356                                //this happens when the new alignment crosses the protein terminus
357                                if( optAln[block][1][pos-1]+cp<ca2len &&
358                                                optAln[block][1][pos]+cp >= ca2len) {
359                                        currBlock = new ArrayList<>(2);
360                                        currBlock.add( new ArrayList<Integer>());
361                                        currBlock.add( new ArrayList<Integer>());
362                                        blocks.add(currBlock);
363                                }
364                                currBlock.get(0).add( optAln[block][0][pos] );
365                                currBlock.get(1).add( (optAln[block][1][pos]+cp ) % ca2len);
366                        }
367                }
368
369                // save permuted blocks to afpChain
370                assignOptAln(afpChain,blocks);
371        }
372
373        /**
374         * Sometimes it's convenient to store an alignment using java collections,
375         * where <tt>blocks.get(blockNum).get(0).get(pos)</tt> specifies the aligned
376         * residue at position <i>pos</i> of block <i>blockNum</i> of the first
377         * protein.
378         *
379         * This method takes such a collection and stores it into the afpChain's
380         * {@link AFPChain#setOptAln(int[][][]) optAln}, setting the associated
381         * length variables as well.
382         *
383         * @param afpChain
384         * @param blocks
385         */
386        private static void assignOptAln(AFPChain afpChain, List<List<List<Integer>>> blocks)
387        {
388
389                int[][][] optAln = new int[blocks.size()][][];
390                int[] optLen = new int[blocks.size()];
391                int optLength = 0;
392                int numBlocks = blocks.size();
393
394                for(int block = 0; block < numBlocks; block++) {
395                        // block should be 2xN rectangular
396                        assert(blocks.get(block).size() == 2);
397                        assert( blocks.get(block).get(0).size() == blocks.get(block).get(1).size());
398
399                        optLen[block] = blocks.get(block).get(0).size();
400                        optLength+=optLen[block];
401
402                        optAln[block] = new int[][] {
403                                        new int[optLen[block]],
404                                        new int[optLen[block]]
405                        };
406                        for(int pos = 0; pos < optLen[block]; pos++) {
407                                optAln[block][0][pos] = blocks.get(block).get(0).get(pos);
408                                optAln[block][1][pos] = blocks.get(block).get(1).get(pos);
409                        }
410                }
411
412
413                afpChain.setBlockNum(numBlocks);
414                afpChain.setOptAln(optAln);
415                afpChain.setOptLen(optLen);
416                afpChain.setOptLength(optLength);
417
418                // TODO I don't know what these do. Should they be set?
419                //afpChain.setBlockSize(blockSize);
420                //afpChain.setBlockResList(blockResList);
421                //afpChain.setChainLen(chainLen);
422
423        }
424
425        /**
426         * Finds the optimal alignment between two proteins allowing for a circular
427         * permutation (CP).
428         * <p>
429         * The precise algorithm is controlled by the
430         * {@link OptimalCECPParameters parameters}. If the parameter
431         * {@link OptimalCECPParameters#isTryAllCPs() tryAllCPs} is true, all possible
432         * CP sites are tried and the optimal site is returned. Otherwise, the
433         * {@link OptimalCECPParameters#getCPPoint() cpPoint} parameter is used to
434         * determine the CP point, greatly reducing the computation required.
435         *
436         * @param ca1 CA atoms of the first protein
437         * @param ca2 CA atoms of the second protein
438         * @param param {@link CeParameters} object
439         * @return The best-scoring alignment
440         * @throws StructureException
441         *
442         * @see #alignOptimal(Atom[], Atom[], Object, AFPChain[])
443         */
444        @Override
445        public AFPChain align(Atom[] ca1, Atom[] ca2, Object param)
446        throws StructureException
447        {
448                if(params.isTryAllCPs()) {
449                        return alignOptimal(ca1,ca2,param,null);
450                } else {
451                        int cpPoint = params.getCPPoint();
452                        return alignPermuted(ca1, ca2, param, cpPoint);
453                }
454        }
455
456        /**
457         * Finds the optimal alignment between two proteins allowing for a circular
458         * permutation (CP).
459         * <p>
460         * This algorithm performs a CE alignment for each possible CP site. This is
461         * quite slow.
462         *
463         * @param ca1 CA atoms of the first protein
464         * @param ca2 CA atoms of the second protein
465         * @param param {@link CeParameters} object
466         * @param alignments If not null, should be an empty array of the same length as
467         *  ca2. This will be filled with the alignments from permuting ca2 by
468         *  0 to n-1 residues.
469         * @return The best-scoring alignment
470         * @throws StructureException
471         */
472        public AFPChain alignOptimal(Atom[] ca1, Atom[] ca2, Object param, AFPChain[] alignments)
473        throws StructureException
474        {
475                long startTime = System.currentTimeMillis();
476
477                if(alignments != null && alignments.length != ca2.length) {
478                        throw new IllegalArgumentException("scores param should have same length as ca2");
479                }
480
481                AFPChain unaligned = super.align(ca1, ca2, param);
482                AFPChain bestAlignment = unaligned;
483
484                if(debug) {
485                        // print progress bar header
486                        System.out.print("|");
487                        for(int cp=1;cp<ca2.length-1;cp++) {
488                                System.out.print("=");
489                        }
490                        System.out.println("|");
491                        System.out.print(".");
492                }
493
494                if(alignments != null) {
495                        alignments[0] = unaligned;
496                }
497
498                for(int cp=1;cp<ca2.length;cp++) {
499                        // clone ca2 to prevent side effects from propegating
500                        Atom[] ca2p = StructureTools.cloneAtomArray(ca2);
501
502                        //permute one each time. Alters ca2p as a side effect
503                        AFPChain currentAlignment = alignPermuted(ca1,ca2p,param,cp);
504
505                        // increment progress bar
506                        if(debug) System.out.print(".");
507
508                        // fix up names, since cloning ca2 wipes it
509
510                        if (ca2.length!=0 && ca2[0].getGroup().getChain()!=null && ca2[0].getGroup().getChain().getStructure()!=null) {
511                                currentAlignment.setName2(ca2[0].getGroup().getChain().getStructure().getName()+" CP="+cp);
512                        }
513
514                        double currentScore = currentAlignment.getAlignScore();
515
516                        if(alignments != null) {
517                                alignments[cp] = currentAlignment;
518                        }
519
520                        if(currentScore>bestAlignment.getAlignScore()) {
521                                bestAlignment = currentAlignment;
522                        }
523                }
524                if(debug) {
525                        long elapsedTime = System.currentTimeMillis()-startTime;
526                        System.out.println();
527                        System.out.format("%d alignments took %.4f s (%.1f ms avg)\n",
528                                        ca2.length, elapsedTime/1000., (double)elapsedTime/ca2.length);
529                }
530
531
532                return bestAlignment;
533
534        }
535
536
537
538
539        public static void main(String[] args){
540                try {
541                        String name1, name2;
542
543                        //int[] cps= new int[] {};
544
545                        //Concanavalin
546                        //name1 = "2pel.A";
547                        //name2 = "3cna";
548                        //cps = new int[] {122,0,3};
549
550                        //small case
551                        name1 = "d1qdmA1";
552                        //name1 = "1QDM.A";
553                        name2 = "d1nklA_";
554                        /*cps = new int[] {
555                                        //41, // CECP optimum
556                                        19,59, // unpermuted local minima in TM-score
557                                        //39, // TM-score optimum
558                                        0,
559                        };*/
560
561                        //1itb selfsymmetry
562                        //name1 = "1ITB.A";
563                        //name2 = "1ITB.A";
564                        //cps = new int[] {92};
565
566                        name1=name2 = "2LSQ";
567
568                        OptimalCECPMain ce = new OptimalCECPMain();
569                        CeParameters params = (CeParameters) ce.getParameters();
570                        ce.setParameters(params);
571
572                        AtomCache cache = new AtomCache();
573
574                        Atom[] ca1 = cache.getAtoms(name1);
575                        Atom[] ca2 = cache.getAtoms(name2);
576
577                        AFPChain afpChain;
578
579
580
581                        // find optimal solution
582                        AFPChain[] alignments = new AFPChain[ca2.length];
583                        afpChain = ce.alignOptimal(ca1, ca2, params, alignments);
584                        System.out.format("Optimal Score: %.2f\n", afpChain.getAlignScore());
585
586                        System.out.println("Pos\tScore\tTMScore\tLen\tRMSD\tBlocks");
587                        for(int i = 0; i< alignments.length; i++) {
588                                double tm = AFPChainScorer.getTMScore(alignments[i], ca1, ca2);
589                                System.out.format("%d\t%.2f\t%.2f\t%d\t%.2f\t%d\n",
590                                                i,
591                                                alignments[i].getAlignScore(),
592                                                tm,
593                                                alignments[i].getOptLength(),
594                                                alignments[i].getTotalRmsdOpt(),
595                                                alignments[i].getBlockNum()
596                                );
597                        }
598
599                        //displayAlignment(afpChain,ca1,ca2);
600
601                        // permuted alignment
602                        //for(int cp : cps) {
603                                // new copy of ca2, since alignPermuted has side effects
604                                //Atom[] ca2clone = cache.getAtoms(name2);
605                                //afpChain = ce.alignPermuted(ca1, ca2clone, params, cp);
606                                //displayAlignment(afpChain, ca1, ca2);
607
608                                //displayAlignment(alignments[cp],ca1,ca2);
609                        //}
610
611                        // CECP alignment
612                        CeCPMain cecp = new CeCPMain();
613                        afpChain = cecp.align(ca1, ca2);
614                        displayAlignment(afpChain,ca1,ca2);
615
616                        System.out.println("Inspect additional alignments?");
617                        Scanner scanner = new Scanner(System.in);
618                        System.out.print("CP location [0,"+ca2.length+"): ");
619                        while(scanner.hasNext()) {
620                                if(scanner.hasNextInt()) {
621                                        int cp = scanner.nextInt();
622
623                                        if(0<=cp && cp<ca2.length) {
624                                                alignments[cp].setName1(name1);
625                                                alignments[cp].setName2(name2+"@"+cp);
626                                                displayAlignment(alignments[cp],ca1,ca2);
627                                                Thread.sleep(1000);
628                                        }
629                                } else {
630                                        //String next = scanner.nextLine();
631                                }
632                                System.out.print("CP location [0,"+ca2.length+"): ");
633                        }
634                        scanner.close();
635
636                } catch (Exception e) {
637                        e.printStackTrace();
638                }
639        }
640
641        /**
642         * Try showing a the afpChain in a GUI.
643         *
644         * <p>requires additional dependencies biojava-structure-gui and JmolApplet
645         *
646         * @param afpChain
647         * @param ca1
648         * @param ca2
649         * @throws ClassNotFoundException
650         * @throws NoSuchMethodException
651         * @throws InvocationTargetException
652         * @throws IllegalAccessException
653         * @throws StructureException
654         */
655        private static void displayAlignment(AFPChain afpChain, Atom[] ca1, Atom[] ca2) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, StructureException {
656                Atom[] ca1clone = StructureTools.cloneAtomArray(ca1);
657                Atom[] ca2clone = StructureTools.cloneAtomArray(ca2);
658                if (! GuiWrapper.isGuiModuleInstalled()) {
659                        System.err.println("The biojava-structure-gui and/or JmolApplet modules are not installed. Please install!");
660                        // display alignment in console
661                        System.out.println(afpChain.toCE(ca1clone, ca2clone));
662                } else {
663                        Object jmol = GuiWrapper.display(afpChain,ca1clone,ca2clone);
664                        GuiWrapper.showAlignmentImage(afpChain, ca1clone,ca2clone,jmol);
665                }
666        }
667}