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.survival.cox;
022
023import org.biojava.nbio.survival.cox.stats.ChiSq;
024import org.biojava.nbio.survival.kaplanmeier.figure.ExpressionFigure;
025import org.biojava.nbio.survival.kaplanmeier.figure.KaplanMeierFigure;
026
027import java.text.DecimalFormat;
028import java.util.ArrayList;
029import java.util.LinkedHashMap;
030
031//import org.biojava.nbio.survival.cox.comparators.SurvivalInfoComparator;
032
033/**
034 * Holds the results of a cox analysis where calling dump(), toString() will give an output similar to R
035 * @author Scooter Willis 
036 */
037public class CoxInfo {
038
039        private WaldTestInfo waldTestInfo = null;
040        String message = "";
041        Integer maxIterations = null;
042        Double eps = null;
043        Double toler = null;
044        CoxMethod method;
045        private double[][] imat = null; //:the variance matrix at beta=final
046        private double[][] naive_imat = null; //the original variance matrix used in residuals
047        double[] u = new double[0]; //:score vector
048        int iterations = 0; //:actual number of iterations used
049        int flag = 0; // success flag  1000  did not converge 1 to nvar: rank of the solution
050        double logTest = 0;
051        double logTestpval = 0;
052        double loglikInit = 0; //loglik at beta=initial values, at beta=final
053        double loglikFinal = 0;
054        Double scoreLogrankTest;
055        private Double rscore = null; //robust score
056        private Double rscoreLogrankTestpvalue = null;
057        private double degreeFreedom;
058        private Double scoreLogrankTestpvalue;
059        int numSamples = 0;
060        int numEvents = 0;
061        private LinkedHashMap<String, String> metaDataFilter = null;
062        private LinkedHashMap<String, CoxCoefficient> coefficientsList = new LinkedHashMap<>();
063        LinkedHashMap<Double, Double> baselineSurvivorFunction = new LinkedHashMap<>();
064        ArrayList<SurvivalInfo> survivalInfoList = new ArrayList<>();
065        /**
066         *
067         */
068        public KaplanMeierFigure kmf = null;
069        /**
070         *
071         */
072        public ExpressionFigure ef = null;
073
074        /**
075         *
076         * @return
077         */
078        public ArrayList<SurvivalInfo> getSurvivalInfoList() {
079                return survivalInfoList;
080        }
081
082        /**
083         *
084         * @param var
085         * @throws Exception
086         */
087        public void setVariance(double[][] var) throws Exception {
088
089                //   if (Math.abs(var[0][1] - var[1][0]) > .0000000000001) { //in the CoxCC correction looks like a precision error keeps these from being equal 10-19
090                //       throw new Exception("Expecting diagonal to be equal");
091                //   }
092                imat = new double[var.length][var[0].length];
093                for (int i = 0; i < var.length; i++) {
094                        for (int j = 0; j < var[0].length; j++) {
095                                imat[i][j] = var[i][j];
096                        }
097                }
098                calcSummaryValues();
099        }
100
101        /**
102         *
103         * @return
104         */
105        public double[][] getVariance() {
106                double[][] var = new double[imat.length][imat[0].length];
107                for (int i = 0; i < var.length; i++) {
108                        for (int j = 0; j < var[0].length; j++) {
109                                var[i][j] = imat[i][j];
110                        }
111                }
112
113                return var;
114        }
115
116        /**
117         *
118         * @param var
119         * @throws Exception
120         */
121        public void setNaiveVariance(double[][] var) throws Exception {
122//        if (var[0][1] != var[1][0]) {
123//            throw new Exception("Expecting diagonal to be equal");
124//        }
125
126
127                naive_imat = new double[var.length][var[0].length];
128                for (int i = 0; i < var.length; i++) {
129                        for (int j = 0; j < var[0].length; j++) {
130                                naive_imat[i][j] = var[i][j];
131                        }
132                }
133
134                calcSummaryValues();
135        }
136
137        /**
138         *
139         * @return
140         */
141        public double[][] getNaiveVariance() {
142                double[][] var = new double[imat.length][imat[0].length];
143                for (int i = 0; i < var.length; i++) {
144                        for (int j = 0; j < var[0].length; j++) {
145                                var[i][j] = naive_imat[i][j];
146                        }
147                }
148
149                return var;
150
151        }
152
153        /**
154         *
155         * @param data
156         */
157        public void setSurvivalInfoList(ArrayList<SurvivalInfo> data) {
158                survivalInfoList = data;
159                numSamples = data.size();
160
161                for (SurvivalInfo si : data) {
162                        if (si.getStatus() == 1) {
163                                numEvents++;
164                        }
165                }
166        }
167
168        /**
169         *
170         * @return
171         */
172        public double[] getWeighted() {
173                double[] weighted = new double[survivalInfoList.size()];
174                int p = 0;
175                for (SurvivalInfo si : this.survivalInfoList) {
176                        weighted[p] = si.getWeight();
177                        p++;
178                }
179                return weighted;
180        }
181
182        /**
183         *
184         * @return
185         */
186        public double[][] getVariableResiduals() {
187                ArrayList<String> variables = new ArrayList<>(coefficientsList.keySet());
188                double[][] rr = new double[survivalInfoList.size()][variables.size()];
189                int p = 0;
190                for (SurvivalInfo si : this.survivalInfoList) {
191                        int i = 0;
192                        for (String v : variables) {
193                                rr[p][i] = si.getResidualVariable(v);
194                                i++;
195                        }
196                        p++;
197                }
198
199                return rr;
200        }
201
202        /**
203         *
204         * @param rr
205         */
206        public void setVariableResiduals(double[][] rr) {
207                ArrayList<String> variables = new ArrayList<>(coefficientsList.keySet());
208
209                int p = 0;
210                for (SurvivalInfo si : this.survivalInfoList) {
211                        int i = 0;
212                        for (String v : variables) {
213                                si.setResidualVariable(v, rr[p][i]);
214                                i++;
215                        }
216                        p++;
217                }
218
219        }
220
221        /**
222         *
223         * @return
224         */
225        public int getNumberCoefficients() {
226                return coefficientsList.size();
227        }
228
229        /**
230         *
231         * @param name
232         * @return
233         */
234        public CoxCoefficient getCoefficient(String name) {
235                return coefficientsList.get(name);
236        }
237
238        /**
239         *
240         * @param name
241         * @param coefficient
242         */
243        public void setCoefficient(String name, CoxCoefficient coefficient) {
244                coefficientsList.put(name, coefficient);
245        }
246
247        /**
248         *
249         * @param header
250         * @param beginLine
251         * @param beginCell
252         * @param endCell
253         * @param endLine
254         * @return
255         */
256        public String getCoefficientText(boolean header, String beginLine, String beginCell, String endCell, String endLine) {
257                String o = "";
258                if (header) {
259                        String robust = "";
260                        if (naive_imat != null) {
261                                robust = beginCell + "robust se" + endCell;
262                        }
263                        o = o + beginLine + beginCell + fmtpl("", 9) + endCell + beginCell + fmtpl("coef", 9) + endCell + beginCell + fmtpl("se(coef)", 9) + endCell + robust + beginCell + fmtpl("z", 9) + endCell + beginCell + fmtpl("p-value", 9) + endCell + beginCell + fmtpl("HR", 9) + endCell + beginCell + fmtpl("lower .95", 9) + endCell + beginCell + fmtpl("upper .95", 9) + endCell + endLine;
264                }//Coefficients,Coe,StdErr,HR,p-value,HR Lo 95%,HR Hi 95%
265
266                for (CoxCoefficient coe : coefficientsList.values()) {
267                        String robust = "";
268                        String stderror = "";
269                        if (naive_imat != null) {
270                                stderror = beginCell + fmt(coe.getRobustStdError(), 5, 9) + endCell;
271                                robust = beginCell + fmt(coe.getStdError(), 5, 9) + endCell;
272                        } else {
273                                stderror = beginCell + fmt(coe.getStdError(), 5, 9) + endCell;
274                        }
275                        o = o + beginLine + beginCell + fmtpr(coe.getName(), 9) + endCell + beginCell + fmt(coe.getCoeff(), 5, 9) + stderror + robust + endCell + beginCell + fmt(coe.getZ(), 5, 9) + endCell + beginCell + fmt(coe.getPvalue(), 6, 9) + endCell + beginCell + fmt(coe.getHazardRatio(), 3, 9) + endCell + beginCell + fmt(coe.getHazardRatioLoCI(), 3, 9) + endCell + beginCell + fmt(coe.getHazardRatioHiCI(), 3, 9) + endCell + endLine;
276                }
277                return o;
278        }
279
280        /**
281         *
282         * @param d
283         * @param precision
284         * @param pad
285         * @return
286         */
287        public static String fmt(Double d, int precision, int pad) {
288                if(d == null)
289                        return "";
290                if(Double.isNaN(d))
291                        return "";
292                String value = "";
293                DecimalFormat dfe = new DecimalFormat("0.00E0");
294                String dpad = "0.";
295                double p = 1.0;
296                for (int i = 0; i < (precision); i++) {
297                        dpad = dpad + "0";
298                        p = p / 10.0;
299                }
300                DecimalFormat df = new DecimalFormat(dpad);
301                if (Math.abs(d) >= p) {
302                        value = df.format(d);
303                } else {
304                        value = dfe.format(d);
305                }
306                int length = value.length();
307                int extra = pad - length;
308                if (extra > 0) {
309                        for (int i = 0; i < extra; i++) {
310                                value = " " + value;
311                        }
312                }
313                return value;
314        }
315
316        /**
317         *
318         */
319        private void calcSummaryValues() {
320
321                //beta
322
323                ArrayList<String> variables = new ArrayList<>(coefficientsList.keySet());
324                for (int i = 0; i < variables.size(); i++) {
325                        String variable = variables.get(i);
326                        CoxCoefficient coe = coefficientsList.get(variable);
327                        coe.setStdError(Math.sqrt(imat[i][i])); //values can be updated to reflect new error
328                        if (naive_imat != null) {
329                                coe.setRobustStdError(Math.sqrt(naive_imat[i][i]));
330                        }
331                        coe.setZ(coe.getCoeff() / coe.getStdError());
332                        coe.setPvalue(ChiSq.norm(Math.abs(coe.getCoeff() / coe.getStdError())));
333                        //z <- qnorm((1 + conf.int)/2, 0, 1)
334                        double z = 1.959964;
335                        coe.setHazardRatioLoCI(Math.exp(coe.getCoeff() - z * coe.getStdError()));
336                        coe.setHazardRatioHiCI(Math.exp(coe.getCoeff() + z * coe.getStdError()));
337                }
338
339                logTest = -2 * (loglikInit - loglikFinal);
340                logTestpval = ChiSq.chiSq(logTest, (int) degreeFreedom);
341
342                scoreLogrankTestpvalue = ChiSq.chiSq(scoreLogrankTest, (int) degreeFreedom);
343                if (rscore != null) {
344                        rscoreLogrankTestpvalue = ChiSq.chiSq(rscore, (int) degreeFreedom);
345                }
346        }
347
348        /**
349         *
350         */
351        public void dump() {
352
353                //need an ordered list for comparing to R dumps
354
355//        ArrayList<SurvivalInfo> orderedSurvivalInfoList = new ArrayList<SurvivalInfo>(survivalInfoList);
356//        SurvivalInfoComparator sicSort = new SurvivalInfoComparator();
357                //       Collections.sort(orderedSurvivalInfoList,sicSort);
358
359
360                System.out.println();
361                System.out.println("$coef");
362                for (CoxCoefficient coe : coefficientsList.values()) {
363                        System.out.print(coe.getCoeff() + " ");
364                }
365                System.out.println();
366                System.out.println("$means");
367
368                for (CoxCoefficient coe : coefficientsList.values()) {
369                        System.out.print(coe.getMean() + " ");
370                }
371                System.out.println();
372                System.out.println("$u");
373
374                for (double d : u) {
375                        System.out.print(d + " ");
376                }
377
378                System.out.println();
379                System.out.println("$imat");
380                for (int i = 0; i < imat.length; i++) {
381                        for (int j = 0; j < imat[0].length; j++) {
382                                System.out.print(imat[i][j] + " ");
383                        }
384                        System.out.println();
385                }
386
387                if (this.naive_imat != null) {
388                        System.out.println("$naive_imat");
389                        for (int i = 0; i < naive_imat.length; i++) {
390                                for (int j = 0; j < naive_imat[0].length; j++) {
391                                        System.out.print(naive_imat[i][j] + " ");
392                                }
393                                System.out.println();
394                        }
395                }
396
397                System.out.println();
398                System.out.println("$loglik");
399
400                System.out.println(loglikInit + " " + loglikFinal);
401
402                System.out.println();
403                System.out.println("$sctest");
404
405                System.out.println(this.scoreLogrankTest);
406
407                System.out.println("$iter");
408                System.out.println(this.iterations);
409
410                System.out.println("$flag");
411                System.out.println(flag);
412
413                System.out.println();
414//        if (false) {
415//            System.out.println("ID      LP       Score      Residuals");
416//            for (SurvivalInfo si : orderedSurvivalInfoList) {
417//                System.out.println(si.getOrder() + " " + si.getLinearPredictor() + " " + si.getScore() + " " + si.getResidual());
418//
419//            }
420//            System.out.println();
421//            ArrayList<String> variables = new ArrayList<String>(coefficientsList.keySet());
422//            System.out.print("Sample");
423//            for (String v : variables) {
424//                System.out.print("    " + v);
425//            }
426//            System.out.println("rr");
427//            for (SurvivalInfo si : orderedSurvivalInfoList) {
428//                System.out.print(si.getOrder());
429//                for (String v : variables) {
430//                    System.out.print("   " + si.getResidualVariable(v));
431//                }
432//                System.out.println();
433//            }
434//        }
435
436        }
437
438        /**
439         *
440         * @param d
441         * @param pad
442         * @return
443         */
444        public String fmtpr(String d, int pad) {
445                int length = d.length();
446                int extra = pad - length;
447                if (extra < 0) {
448                        extra = 0;
449                }
450                String v = d;
451                for (int i = 0; i < extra; i++) {
452                        v = v + " ";
453                }
454                return v;
455        }
456
457        /**
458         * Pad left a string with spaces
459         *
460         * @param d
461         * @param pad
462         * @return
463         */
464        public String fmtpl(String d, int pad) {
465                int length = d.length();
466                int extra = pad - length;
467                if (extra < 0) {
468                        extra = 0;
469                }
470                String v = d;
471                for (int i = 0; i < extra; i++) {
472                        v = " " + v;
473                }
474                return v;
475        }
476
477        @Override
478        public String toString() {
479                return toString("", " ", "\r\n");
480        }
481
482        /**
483         *
484         * @param beginLine
485         * @param del
486         * @param endLine
487         * @return
488         */
489        public String toString(String beginLine, String del, String endLine) {
490
491
492
493                String o = beginLine + fmtpl("", 9) + fmtpl("Avg", 9) + fmtpl("SD", 9) + endLine;
494                for (CoxCoefficient coe : coefficientsList.values()) {
495                        o = o + beginLine + fmtpr(coe.getName(), 9) + fmt(coe.getMean(), 4, 9) + fmt(coe.getStandardDeviation(), 4, 9) + endLine;
496
497                }
498
499                o = o + beginLine + endLine;
500
501
502
503                o = o + beginLine + "n= " + this.numSamples + ", number of events=" + this.numEvents + endLine;
504                o = o + getCoefficientText(true, beginLine, del, "", endLine);
505
506                o = o + beginLine + endLine;
507
508                if (baselineSurvivorFunction.size() > 0) {
509                        o = o + beginLine + "Baseline Survivor Function (at predictor means)" + endLine;
510                        for (Double time : baselineSurvivorFunction.keySet()) {
511                                Double mean = baselineSurvivorFunction.get(time);
512                                o = o + beginLine + fmt(time, 4, 10) + fmt(mean, 4, 10) + endLine;
513                        }
514                }
515
516                o = o + beginLine + endLine;
517                o = o + beginLine + "Overall Model Fit" + endLine;
518                o = o + beginLine + "Iterations=" + iterations + endLine;
519
520
521                o = o + beginLine + "Likelihood ratio test = " + fmt(this.logTest, 2, 0) + " df=" + this.degreeFreedom + " p-value=" + fmt(this.logTestpval, 7, 0) + endLine;
522
523                o = o + beginLine + "Wald test             = " + fmt(waldTestInfo.getTest(), 2, 0) + " df=" + waldTestInfo.getDf() + " p-value=" + fmt(waldTestInfo.getPvalue(), 7, 0) + endLine;
524                o = o + beginLine + "Score (logrank) test  = " + fmt(scoreLogrankTest, 2, 0) + " df=" + ((int) (this.degreeFreedom)) + " p-value=" + fmt(this.scoreLogrankTestpvalue, 7, 0);
525
526                if (this.rscore != null) {
527                        o = o + ",  Robust = " + fmt(rscore, 2, 0) + " p-value=" + fmt(rscoreLogrankTestpvalue, 7, 0);
528
529                }
530
531                o = o + endLine;
532
533                //       o = o + "Rank of solution flag=" + flag + "\r\n";
534                //       o = o + "Log lik Initial=" + loglikInit + "\r\n";
535                //       o = o + "Log lik Final=" + loglikFinal + "\r\n";
536                o = o + beginLine + "Method=" + method.name() + endLine;
537
538
539
540
541                return o;
542        }
543
544        /**
545         * @return the scoreLogrankTest
546         */
547        public double getChiSquare() {
548                return scoreLogrankTest;
549        }
550
551        /**
552         * @return the degreeFreedom
553         */
554        public double getDegreeFreedom() {
555                return degreeFreedom;
556        }
557
558        /**
559         * @return the scoreLogrankTestpvalue
560         */
561        public double getOverallModelFitPvalue() {
562                return scoreLogrankTestpvalue;
563        }
564
565        /**
566         * @return the rscore
567         */
568        public Double getRscore() {
569                return rscore;
570        }
571
572        /**
573         * @param rscore the rscore to set
574         */
575        public void setRscore(Double rscore) {
576                this.rscore = rscore;
577                if (rscore != null) {
578                        rscoreLogrankTestpvalue = ChiSq.chiSq(rscore, (int) degreeFreedom);
579                }
580        }
581
582        /**
583         * @return the rscoreLogrankTestpvalue
584         */
585        public Double getRscoreLogrankTestpvalue() {
586                return rscoreLogrankTestpvalue;
587        }
588
589        /**
590         * @param rscoreLogrankTestpvalue the rscoreLogrankTestpvalue to set
591         */
592        public void setRscoreLogrankTestpvalue(Double rscoreLogrankTestpvalue) {
593                this.rscoreLogrankTestpvalue = rscoreLogrankTestpvalue;
594        }
595
596        /**
597         * @return the scoreLogrankTest
598         */
599        public Double getScoreLogrankTest() {
600                return scoreLogrankTest;
601        }
602
603        /**
604         * @param scoreLogrankTest the scoreLogrankTest to set
605         */
606        public void setScoreLogrankTest(Double scoreLogrankTest) {
607                this.scoreLogrankTest = scoreLogrankTest;
608        }
609
610        /**
611         * @return the scoreLogrankTestpvalue
612         */
613        public Double getScoreLogrankTestpvalue() {
614                return scoreLogrankTestpvalue;
615        }
616
617        /**
618         * @param scoreLogrankTestpvalue the scoreLogrankTestpvalue to set
619         */
620        public void setScoreLogrankTestpvalue(Double scoreLogrankTestpvalue) {
621                this.scoreLogrankTestpvalue = scoreLogrankTestpvalue;
622        }
623
624        /**
625         * @return the metaDataFilter
626         */
627        public LinkedHashMap<String, String> getMetaDataFilter() {
628                return metaDataFilter;
629        }
630
631        /**
632         * @param metaDataFilter the metaDataFilter to set
633         */
634        public void setMetaDataFilter(LinkedHashMap<String, String> metaDataFilter) {
635                this.metaDataFilter = metaDataFilter;
636        }
637
638        /**
639         * @return the coefficientsList
640         */
641        public LinkedHashMap<String, CoxCoefficient> getCoefficientsList() {
642                return coefficientsList;
643        }
644
645        /**
646         * @return the waldTestInfo
647         */
648        public WaldTestInfo getWaldTestInfo() {
649                return waldTestInfo;
650        }
651
652        /**
653         * @return the imat
654         */
655        public double[][] getImat() {
656                return imat;
657        }
658
659        /**
660         * @return the naive_imat
661         */
662        public double[][] getNaive_imat() {
663                return naive_imat;
664        }
665
666        /**
667         * @param degreeFreedom the degreeFreedom to set
668         */
669        public void setDegreeFreedom(double degreeFreedom) {
670                this.degreeFreedom = degreeFreedom;
671        }
672
673        /**
674         * @param waldTestInfo the waldTestInfo to set
675         */
676        public void setWaldTestInfo(WaldTestInfo waldTestInfo) {
677                this.waldTestInfo = waldTestInfo;
678        }
679}