// @(#)root/roostats:$Id$
// Author: Kyle Cranmer, Sven Kreiss   23/05/10
/*************************************************************************
 * Copyright (C) 1995-2008, Rene Brun and Fons Rademakers.               *
 * All rights reserved.                                                  *
 *                                                                       *
 * For the licensing terms see $ROOTSYS/LICENSE.                         *
 * For the list of contributors see $ROOTSYS/README/CREDITS.             *
 *************************************************************************/

/**
Same purpose as HybridCalculatorOriginal, but different implementation.
*/

#include "RooStats/HybridCalculator.h"
#include "RooStats/ToyMCSampler.h"


ClassImp(RooStats::HybridCalculator)

using namespace RooStats;
using namespace std;



int HybridCalculator::CheckHook(void) const {

   if( fPriorNuisanceNull && (!fNullModel->GetNuisanceParameters() || fNullModel->GetNuisanceParameters()->getSize() == 0) ) {
      oocoutE((TObject*)0,InputArguments)  << "HybridCalculator - Nuisance PDF has been specified, but is unaware of which parameters are the nuisance parameters. Must set nuisance parameters in the Null ModelConfig." << endl;
      return -1; // error
   }
   if( fPriorNuisanceAlt && (!fAltModel->GetNuisanceParameters() || fAltModel->GetNuisanceParameters()->getSize() == 0) ) {
      oocoutE((TObject*)0,InputArguments)  << "HybridCalculator - Nuisance PDF has been specified, but is unaware of which parameters are the nuisance parameters. Must set nuisance parameters in the Alt ModelConfig" << endl;
      return -1; // error
   }

   return 0; // ok
}


int HybridCalculator::PreNullHook(RooArgSet* /*parameterPoint*/, double obsTestStat) const {


   // ****** any TestStatSampler ********

   if(fPriorNuisanceNull) {
      // Setup Priors for ad hoc Hybrid
      fTestStatSampler->SetPriorNuisance(fPriorNuisanceNull);
   } else if(
      fNullModel->GetNuisanceParameters() == NULL ||
      fNullModel->GetNuisanceParameters()->getSize() == 0
   ) {
      oocoutI((TObject*)0,InputArguments)
       << "HybridCalculator - No nuisance parameters specified for Null model and no prior forced. "
       << "Case is reduced to simple hypothesis testing with no uncertainty." << endl;
   } else {
      oocoutI((TObject*)0,InputArguments) << "HybridCalculator - Using uniform prior on nuisance parameters (Null model)." << endl;
   }


   // ***** ToyMCSampler specific *******

   // check whether TestStatSampler is a ToyMCSampler
   ToyMCSampler *toymcs = dynamic_cast<ToyMCSampler*>(GetTestStatSampler());
   if(toymcs) {
      oocoutI((TObject*)0,InputArguments) << "Using a ToyMCSampler. Now configuring for Null." << endl;

      // variable number of toys
      if(fNToysNull >= 0) toymcs->SetNToys(fNToysNull);

      // adaptive sampling
      if(fNToysNullTail) {
         oocoutI((TObject*)0,InputArguments) << "Adaptive Sampling" << endl;
         if(GetTestStatSampler()->GetTestStatistic()->PValueIsRightTail()) {
            toymcs->SetToysRightTail(fNToysNullTail, obsTestStat);
         }else{
            toymcs->SetToysLeftTail(fNToysNullTail, obsTestStat);
         }
      }else{
         toymcs->SetToysBothTails(0, 0, obsTestStat); // disable adaptive sampling
      }

      GetNullModel()->LoadSnapshot();
   }

   return 0;
}


int HybridCalculator::PreAltHook(RooArgSet* /*parameterPoint*/, double obsTestStat) const {

   // ****** any TestStatSampler ********

   if(fPriorNuisanceAlt){
     // Setup Priors for ad hoc Hybrid
     fTestStatSampler->SetPriorNuisance(fPriorNuisanceAlt);
   } else if (
      fAltModel->GetNuisanceParameters()==NULL ||
      fAltModel->GetNuisanceParameters()->getSize()==0
   ) {
      oocoutI((TObject*)0,InputArguments)
       << "HybridCalculator - No nuisance parameters specified for Alt model and no prior forced. "
       << "Case is reduced to simple hypothesis testing with no uncertainty." << endl;
   } else {
      oocoutI((TObject*)0,InputArguments) << "HybridCalculator - Using uniform prior on nuisance parameters (Alt model)." << endl;
   }


   // ***** ToyMCSampler specific *******

   // check whether TestStatSampler is a ToyMCSampler
   ToyMCSampler *toymcs = dynamic_cast<ToyMCSampler*>(GetTestStatSampler());
   if(toymcs) {
      oocoutI((TObject*)0,InputArguments) << "Using a ToyMCSampler. Now configuring for Alt." << endl;

      // variable number of toys
      if(fNToysAlt >= 0) toymcs->SetNToys(fNToysAlt);

      // adaptive sampling
      if(fNToysAltTail) {
         oocoutI((TObject*)0,InputArguments) << "Adaptive Sampling" << endl;
         if(GetTestStatSampler()->GetTestStatistic()->PValueIsRightTail()) {
            toymcs->SetToysLeftTail(fNToysAltTail, obsTestStat);
         }else{
            toymcs->SetToysRightTail(fNToysAltTail, obsTestStat);
         }
      }else{
         toymcs->SetToysBothTails(0, 0, obsTestStat); // disable adaptive sampling
      }
   }

   return 0;
}




 HybridCalculator.cxx:1
 HybridCalculator.cxx:2
 HybridCalculator.cxx:3
 HybridCalculator.cxx:4
 HybridCalculator.cxx:5
 HybridCalculator.cxx:6
 HybridCalculator.cxx:7
 HybridCalculator.cxx:8
 HybridCalculator.cxx:9
 HybridCalculator.cxx:10
 HybridCalculator.cxx:11
 HybridCalculator.cxx:12
 HybridCalculator.cxx:13
 HybridCalculator.cxx:14
 HybridCalculator.cxx:15
 HybridCalculator.cxx:16
 HybridCalculator.cxx:17
 HybridCalculator.cxx:18
 HybridCalculator.cxx:19
 HybridCalculator.cxx:20
 HybridCalculator.cxx:21
 HybridCalculator.cxx:22
 HybridCalculator.cxx:23
 HybridCalculator.cxx:24
 HybridCalculator.cxx:25
 HybridCalculator.cxx:26
 HybridCalculator.cxx:27
 HybridCalculator.cxx:28
 HybridCalculator.cxx:29
 HybridCalculator.cxx:30
 HybridCalculator.cxx:31
 HybridCalculator.cxx:32
 HybridCalculator.cxx:33
 HybridCalculator.cxx:34
 HybridCalculator.cxx:35
 HybridCalculator.cxx:36
 HybridCalculator.cxx:37
 HybridCalculator.cxx:38
 HybridCalculator.cxx:39
 HybridCalculator.cxx:40
 HybridCalculator.cxx:41
 HybridCalculator.cxx:42
 HybridCalculator.cxx:43
 HybridCalculator.cxx:44
 HybridCalculator.cxx:45
 HybridCalculator.cxx:46
 HybridCalculator.cxx:47
 HybridCalculator.cxx:48
 HybridCalculator.cxx:49
 HybridCalculator.cxx:50
 HybridCalculator.cxx:51
 HybridCalculator.cxx:52
 HybridCalculator.cxx:53
 HybridCalculator.cxx:54
 HybridCalculator.cxx:55
 HybridCalculator.cxx:56
 HybridCalculator.cxx:57
 HybridCalculator.cxx:58
 HybridCalculator.cxx:59
 HybridCalculator.cxx:60
 HybridCalculator.cxx:61
 HybridCalculator.cxx:62
 HybridCalculator.cxx:63
 HybridCalculator.cxx:64
 HybridCalculator.cxx:65
 HybridCalculator.cxx:66
 HybridCalculator.cxx:67
 HybridCalculator.cxx:68
 HybridCalculator.cxx:69
 HybridCalculator.cxx:70
 HybridCalculator.cxx:71
 HybridCalculator.cxx:72
 HybridCalculator.cxx:73
 HybridCalculator.cxx:74
 HybridCalculator.cxx:75
 HybridCalculator.cxx:76
 HybridCalculator.cxx:77
 HybridCalculator.cxx:78
 HybridCalculator.cxx:79
 HybridCalculator.cxx:80
 HybridCalculator.cxx:81
 HybridCalculator.cxx:82
 HybridCalculator.cxx:83
 HybridCalculator.cxx:84
 HybridCalculator.cxx:85
 HybridCalculator.cxx:86
 HybridCalculator.cxx:87
 HybridCalculator.cxx:88
 HybridCalculator.cxx:89
 HybridCalculator.cxx:90
 HybridCalculator.cxx:91
 HybridCalculator.cxx:92
 HybridCalculator.cxx:93
 HybridCalculator.cxx:94
 HybridCalculator.cxx:95
 HybridCalculator.cxx:96
 HybridCalculator.cxx:97
 HybridCalculator.cxx:98
 HybridCalculator.cxx:99
 HybridCalculator.cxx:100
 HybridCalculator.cxx:101
 HybridCalculator.cxx:102
 HybridCalculator.cxx:103
 HybridCalculator.cxx:104
 HybridCalculator.cxx:105
 HybridCalculator.cxx:106
 HybridCalculator.cxx:107
 HybridCalculator.cxx:108
 HybridCalculator.cxx:109
 HybridCalculator.cxx:110
 HybridCalculator.cxx:111
 HybridCalculator.cxx:112
 HybridCalculator.cxx:113
 HybridCalculator.cxx:114
 HybridCalculator.cxx:115
 HybridCalculator.cxx:116
 HybridCalculator.cxx:117
 HybridCalculator.cxx:118
 HybridCalculator.cxx:119
 HybridCalculator.cxx:120
 HybridCalculator.cxx:121
 HybridCalculator.cxx:122
 HybridCalculator.cxx:123
 HybridCalculator.cxx:124
 HybridCalculator.cxx:125
 HybridCalculator.cxx:126
 HybridCalculator.cxx:127
 HybridCalculator.cxx:128
 HybridCalculator.cxx:129
 HybridCalculator.cxx:130
 HybridCalculator.cxx:131
 HybridCalculator.cxx:132
 HybridCalculator.cxx:133
 HybridCalculator.cxx:134
 HybridCalculator.cxx:135
 HybridCalculator.cxx:136
 HybridCalculator.cxx:137