EWP-BplusToKstMuMu-AngAna/Code/Selection/Utils.hpp

685 lines
27 KiB
C++

//Helper functions
//Yes, they are in a header, because ROOT
//Renata Kopecna
#ifndef UTILS_HPP
#define UTILS_HPP
#include "GlobalFunctions.hh"
#include "Paths.hpp"
#include "MassFit.hpp"
#include <boost/algorithm/string.hpp>
//I HATE ROOT! It's just way easier for root to have functions defined in a header with cross-references.... sigh
//Forward declarations from Paths.hpp
string GetInputFile(string year, string magnet, bool preSelected, bool MC, bool ReferenceChannel, bool PHSP, bool B0, bool K1, bool Inc, bool smallSample);
string GetBDTinputFile(string year, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo);
string GetBDToutputFile(string year, int Run, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo, bool UseLowQ2Range, bool reweighted);
string GetBDToutputFile(int Run, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo, bool UseLowQ2Range, bool reweighted);
string returnFileAddress(string year, int Run, string magnet, bool Preselected, bool BDTed, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo);
/////////////////////////////////////////////
//
// Get basic names and trees
//
/////////////////////////////////////////////
string correct_magnet_string(string magnet){
boost::to_lower(magnet);
if (magnet != "both" && magnet != "up" && magnet != "down"){
coutWarning("Wrong magnet polarity used! Setting magnet to both!");
return magnet = "both";
}
else return magnet;
}
string treeName(bool MC, bool Preselected){
if (MC && Preselected) return "DecayTreeTruthMatched";
else if (!Preselected) return "b2KstKpi0mumuResolvedTuple/DecayTree";
else return "DecayTree";
}
vector <string> get_magnet_vector(string magnet){
magnet = correct_magnet_string(magnet);
if (magnet == "both"){
return {"down","up"};
}
else return {magnet};
}
TChain *get_basic_TChain(string magnet, vector<string> years, bool Preselected, bool MC, bool ReferenceChannel, bool PHSP, bool B0, bool K1, bool Inc){
TChain* tree = new TChain(treeName(MC,Preselected).c_str());
coutInfo("Reading data from TTree... ");
for (auto& yr : years){
for (auto& mag: get_magnet_vector(magnet)){
string path = GetInputFile(yr,mag,Preselected,MC, ReferenceChannel, PHSP, B0, K1, Inc, false);
coutDebug("Adding " + path);
tree->Add(path.c_str()); //TODO: no acces to not-preselected inclusive sample
}
}
return tree;
}
TChain *get_weighted_TChain(vector<string> years, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo){
TChain* tree = new TChain(treeName(MC,true).c_str());
for (auto& yr : years){
tree->Add(GetBDTinputFile(yr,MC,ReferenceChannel,PHSP,KshortDecayInVelo).c_str());
coutDebug("Adding " +GetBDTinputFile(yr,MC,ReferenceChannel,PHSP,KshortDecayInVelo));
}
return tree;
}
TChain *get_BDT_TChain(vector<string> years, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo, bool reweighted){
bool UseLowQ2Range = false;
TChain* tree = new TChain(treeName(MC,true).c_str());
for (auto& yr : years){
tree->Add(GetBDToutputFile(yr,getRunID(yr),MC,ReferenceChannel,PHSP,KshortDecayInVelo,UseLowQ2Range,reweighted).c_str());
coutDebug("Adding " +GetBDToutputFile(yr,getRunID(yr),MC,ReferenceChannel,PHSP,KshortDecayInVelo,UseLowQ2Range,reweighted));
}
return tree;
}
TChain *get_BDT_TChain(string year, int Run, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo, bool reweighted){
bool UseLowQ2Range = false;
vector<string> years;
if (Run == 0) years.push_back(year);
else years = yearsVector(MC,ReferenceChannel,PHSP,Run);
TChain* tree = new TChain(treeName(MC,true).c_str());
for (auto& yr : years){
tree->Add(GetBDToutputFile(yr,getRunID(yr),MC,ReferenceChannel,PHSP,KshortDecayInVelo,UseLowQ2Range,reweighted).c_str());
coutDebug("Adding " +GetBDToutputFile(yr,getRunID(yr),MC,ReferenceChannel,PHSP,KshortDecayInVelo,UseLowQ2Range,reweighted));
}
return tree;
}
TChain *get_FinalOutput_TChain(vector<string> years, bool MC, bool ReferenceChannel, bool PHSP, bool KshortDecayInVelo){ //TODO fix Run vs year
TChain* tree = new TChain(treeName(MC,true).c_str());
for (auto& yr : years){
//tree->Add(GetFinalOutputFile(yr,1,MC,ReferenceChannel,PHSP,KshortDecayInVelo).c_str());
}
return tree;
}
///////////////////////////////////////////////
//
// Cut functions
//
///////////////////////////////////////////////
string getTMcut(bool TM,bool notTM, string customTMbranch, bool gammaTM){
string finalCut = "";
if (TM && notTM){
coutERROR("Cannot cut on TM == 1 && TM == 0! Returning an empty string.");
return "";
}
if (TM){
if (customTMbranch == "") finalCut.append("(" + TMbranch + "==1)");
else finalCut.append("(" + customTMbranch + "==1)");
//Check if the TM branch is not BKGCAT
if (finalCut.find("BKGCAT") == string::npos){
//Gamma category: 1: both passed, 2: one passed, one converted, 3: both converted, 4: one passed, one random, 5: one converted, one random, 6: both failed
if (gammaTM) finalCut.append(" && (" + gammaTMbranch + "<4)");
else finalCut.append(" && (" + gammaTMbranch + "<6)");
}
}
if (notTM){
if (customTMbranch == "") finalCut.append("(" + TMbranch + "==0)");
else finalCut.append("(" + customTMbranch + "==0)");
if (finalCut.find("BKGCAT") == string::npos){
//Gamma category: 1: both passed, 2: one passed, one converted, 3: both converted, 4: one passed, one random, 5: one converted, one random, 6: both failed
if (gammaTM) finalCut.append(" || (" + gammaTMbranch + ">3)");
else finalCut.append(" || (" + gammaTMbranch + ">5)");
}
}
coutTest("TM cut:" + finalCut);
return finalCut;
}
string getJpsicut(){
return "(8.68e6 < Q2 && Q2 < 10.09e6)"; //q2Cuts = "nDiMuonMassBin == 9";
}
string getMuMucut(){
return "(Q2 < 0.98e6 || (1.1e6 < Q2 && Q2 < 8.0e6) || (11.0e6 < Q2 && Q2 < 12.5e6) || Q2 > 15.0e6)";
}
string getFullCut(bool UseOnlyMuMuEvents, bool UseOnlyJpsiEvents, bool SplitInQ2, bool UseLowQ2Range, bool useExtraVar, TMefficiencyClass extraVar,int nExtraBin, double TMVAcut){
string finalCut = "";
string q2Cut = "";
if (UseOnlyMuMuEvents) q2Cut = getMuMucut();
else if (UseOnlyJpsiEvents) q2Cut = getJpsicut();
else if (SplitInQ2){
coutWarning("Using SplitInQ2! Always use with MuMu events!");
if (UseLowQ2Range) q2Cut = "(Q2 < 0.98e6 || (1.1e6 < Q2 && Q2 < 8.0e6))"; //q2Cuts = "nDiMuonMassBin < 9 || nDiMuonMassBin != 2";
else q2Cut = "((11.0e6 < Q2 && Q2 < 12.5e6) || Q2 > 15.0e6)"; //q2Cuts = "nDiMuonMassBin > 9 || nDiMuonMassBin != 12";
}
else q2Cut = "(Q2 >0)"; //dummy cut in order to not having to deal with && at a beggining of the string
finalCut.append(q2Cut);
if (useExtraVar){
string extraVarCut = "";
double lowCut = extraVar.isEquidistant ? extraVar.binEdgesEquidistant.at(nExtraBin) : extraVar.binEdges.at(nExtraBin);
double upperCut = extraVar.isEquidistant ? extraVar.binEdgesEquidistant.at(nExtraBin+1) : extraVar.binEdges.at(nExtraBin+1);
extraVarCut = "(" + to_string(lowCut) + " < " + extraVar.cut+" && " + extraVar.cut+ "<= " + to_string(upperCut) + ")";
coutDebug(extraVarCut);
finalCut.append(" && ");
finalCut.append(extraVarCut);
}
if (TMVAcut > -1.0){
finalCut.append(" && ");
finalCut.append(" (" + string(TMVAmethod) + "response > " + to_string(TMVAcut) + ")");
}
coutTest("Main cut: " + finalCut);
return finalCut ;
}
string getAloneBranch(bool MC, bool TM, string customTMbranch, bool gammaTM){
string customTag = "";
if (MC){
if (!TM) customTag = "NotTM";
else {
if (customTMbranch == "") customTMbranch = TMbranch;
if (customTMbranch == "TMedBKGCAT") customTag = "";
if (customTMbranch == "TMed"){
if (gammaTM) customTag = "_TMed";
else customTag = "_TMed_rndGamma";
}
}
}
return "IsAloneAt" + customTag;
}
string getMultipleCut(bool MC, bool TM, string customTMbranch, bool gammaTM, double TMVAcut){
string aloneBranch = getAloneBranch(MC,TM,customTMbranch, gammaTM);
string cut = (TMVAcut > -1.0) ? string("<=" + to_string(TMVAcut)) : "==0.0";
coutTest("Alone cut: " + aloneBranch + cut);
return aloneBranch + cut;
}
string getFinalCut(bool MC, bool TM, bool notTM, string customTMbranch, bool gammaTM, bool UseOnlyMuMuEvents, bool UseOnlyJpsiEvents, bool SplitInQ2, bool UseLowQ2Range, bool useExtraVar, TMefficiencyClass extraVar,int nExtraBin, double TMVAcut, bool removeMultiple){
string fullCut = getFullCut(UseOnlyMuMuEvents, UseOnlyJpsiEvents, SplitInQ2, UseLowQ2Range, useExtraVar, extraVar, nExtraBin, TMVAcut);
string TMcut = MC ? getTMcut(TM,notTM,customTMbranch,gammaTM) : "";
if (TMcut != "") TMcut = " && (" + TMcut + ")";
string multipleCut = removeMultiple ? string( " && (" +getMultipleCut(MC,TM,customTMbranch,gammaTM,TMVAcut) + ")") : "";
return fullCut + TMcut + multipleCut;
}
//Buggy 2015 MC needs to be removed and replaced by 2016
//TODO: make sure it's not loaded twice everywhere
string checkIf2015MC(string year, bool MC, bool Reference, bool PHSP){
if (Kst2Kpluspi0Resolved && MC && !Reference && !PHSP && year== "2015") return "2016";
else return year;
}
///////////////////////////////////////////////
//
// Sanity checks
//
/////////////////////////////////////////////
bool checkMC(bool &MC, bool ReferenceChannel, bool PHSP, bool mutuallyExclusive){
if (!MC && PHSP){//if PHSP or reference channel, set MC to true
coutWarning("PHSP or Reference channel option selected, setting MC to true");
MC = true;
}
if (mutuallyExclusive && ReferenceChannel && PHSP){
coutERROR("You cannot select both Reference channel and PHSP! Abort.");
return false;
}
return true;
}
bool checkMC(bool ReferenceChannel, bool PHSP){
if (ReferenceChannel && PHSP){
coutERROR("You cannot select both Reference channel and PHSP! Abort.");
return false;
}
return true;
}
bool checkQ2Range(bool UseOnlyJpsi, bool UseOnlyMuMu){
if (UseOnlyJpsi && UseOnlyMuMu){
coutERROR("Make up your mind! Either Jpsi or mumu channel, it cannot be both! Program failed");
return false;
}
return true;
}
bool checkTM(bool MC, bool &TM, bool &nonTM, bool Preselected){
if ( (TM || nonTM) && !MC){
coutWarning("Data is not TruthMatched! Setting TM and nonTM to false!");
TM = false; //Keep nonTM as it is for the MC fit to get the shape
}
if ( (TM || nonTM) && !Preselected){
coutWarning("Stripped MC is not TruthMatched! Setting TM to false!");
TM = false;
nonTM = false;
}
if (TM && nonTM){
coutERROR("Cannot fit TM==1 and TM==0 data at the same time! Make up your mind. Abort.");
return false;
}
return true;
}
bool checkKshort(bool &KshortDecaysInVelo){
if (!Kst2Kspiplus && KshortDecaysInVelo){
coutWarning("No LL/DD tracks in KplusPizero channel! Setting KshortDecaysInVelo to false!");
KshortDecaysInVelo = false;
}
return true;
}
bool checkRefYear(string year){
if (Kst2Kspiplus && (year == "2015" || year =="2017" || year == "2018" )){
coutERROR("Reference channel is only available for 2011, 2012 and 2016!");
return false;
}
if (!Kst2Kspiplus && (year =="2017" || year == "2018" )){
coutERROR("Reference channel is only available for 2011, 2012, 2015 and 2016!");
return false;
}
return true;
}
bool checkEntries(TTree *tree){
Int_t nEvents = tree->GetEntries();
if(nEvents == 0){
coutERROR("Entries in data TTree " + string(tree->GetName()) + " are empty. Abort!");
return false;
}
return true;
}
bool checkYear(int year, bool MC, bool ReferenceChannel, bool PHSP){
bool flag = false;
if (!MC){
if (year == 2011 || year == 2012|| year == 2015 || year == 2016 || year == 2017 || year == 2018 ) flag=true;
else flag=false;
}
else{
if (KshortChannel){ //KS
if (ReferenceChannel){
if (year == 2011 || year == 2012|| year == 2016) flag=true;
else flag=false;
}
else if (PHSP){
if (year == 2011 || year == 2012|| year == 2015 || year == 2016 || year == 2017 || year == 2018 ) flag=true;
else flag=false;
}
else{
if (year == 2011 || year == 2012|| year == 2015 || year == 2016 || year == 2017 || year == 2018 ) flag=true;
else flag=false;
}
}
else{ //K+pi0
if (ReferenceChannel){
if (year == 2011 || year == 2012|| year == 2015 || year == 2016) flag=true;
else flag=false;
}
else if (PHSP){
if (year == 2011 || year == 2012|| year == 2015 || year == 2016 || year == 2017 || year == 2018) flag=true;
else flag=false;
}
else{
if (year == 2011 || year == 2012|| year == 2015 || year == 2016 || year == 2017 || year == 2018) flag=true;
else flag=false;
}
}
}
if (!flag) coutERROR("Wrong year input! Your input is " + to_string(year)+ ". Abort.");
return flag;
}
bool checkRun(int Run){
if(Run != 1 && Run != 2 && Run != 12){
coutERROR("Invalid Run number given: >> " + to_string( Run) + " << . Use only Run 1, Run 2 or Run 12! Exit now.");
return 0;
}
return 1;
}
///////////////////////////////////////////////
//
// Get correct names and tags
//
/////////////////////////////////////////////
string getTMtag(string customTMbranch){
if (customTMbranch == "") customTMbranch = TMbranch;
if (customTMbranch == "TMedBKGCAT") return "_BKGCAT";
if (customTMbranch == "TMed") return "_IDTM";
if (customTMbranch == "TMed_noPi0") return "_noPi0_IDTM";
coutERROR("Wrong customTM branch! Returning string 'wrong'");
return "wrong";
}
string getTMtag(string customTMbranch, bool gammaTM){
if (customTMbranch == "") customTMbranch = TMbranch;
if (customTMbranch == "TMedBKGCAT") return "_BKGCAT";
if (customTMbranch == "TMed") return "_IDTM"+ string(gammaTM ? "" : "_rndGamma");;
if (customTMbranch == "TMed_noPi0") return "_noPi0_IDTM"+ string(gammaTM ? "" : "_rndGamma");;
coutERROR("Wrong customTM branch! Returning string 'wrong'");
return "wrong";
}
string getWeightName(string customTMbranch, bool gammaTM){
if (customTMbranch == "") customTMbranch = TMbranch;
if (customTMbranch == "TMedBKGCAT") return "weight2D_"+firstMCweight;
if (customTMbranch == "TMed") return "weight2D_"+firstMCweight + "_TM" + string(gammaTM ? "" : "_rndGamma");
if (customTMbranch == "TMed_noPi0") return "weight2D_"+firstMCweight + "_noPi0TM";
coutERROR("Wrong customTMbranch name! Returning an empty string.");
return "";
}
std::string getDataTypeTag(bool MC, bool Reference, bool PHSP, bool B0, bool K1, bool Inc){
if (!MC) return "Data";
else{
if (PHSP) return "MC PHSP";
else{
if (B0) return Reference ? "MC B0toKstJpsi" : "MC B0toKstMuMu";
else if (K1) return Reference ? "MC BtoK1Jpsi" : "MC BtoK1MuMu";
else if (Inc) return Reference ? "MC BtoXJpsi" : "MC BtoXMuMu";
else return Reference ? "Reference MC" : "Signal MC";
}
}
}
string getDataTypeTag(bool MC, bool Reference, bool PHSP){
return getDataTypeTag(MC, Reference, PHSP, false, false, false);
}
string getYearRunTag(int Run, string year){
if (Run == 0) return "Year " + year;
else return "Run " + to_string(Run);
}
/////////////////////////////////////////////
//
// Bin edges for a similarly populated bins
//
/////////////////////////////////////////////
void getSimilarlyPopulatedBins(TH1D *histogram, int desiredBins, int nBins){
//rebin (THIS ASSUMES THE ORIGINAL BIN WIDTH IS ONE!!!)
int maxBinContent = histogram->GetEntries()/desiredBins;
TAxis *axis = histogram->GetXaxis();
Double_t xEdges[desiredBins+1];
xEdges[0] = axis->GetBinLowEdge(1);
int i = 0;
for (int b = 1; b <desiredBins+1; b++){
int content = 0;
while (content < maxBinContent){
content = content+histogram->GetBinContent(i);
i++;
}
xEdges[b] = axis->GetBinLowEdge(i);
i++;
}
xEdges[desiredBins] = axis->GetBinUpEdge(nBins);
for (int b = 0; b<desiredBins+1; b++){
cout << xEdges[b] << endl;
}
return;
}
void getSimilarlyPopulatedBinsVar(string var, int nBins, double lowEdge, double highEdge, int desiredBins, string year, string magnet, int Run, bool ReferenceChannel, bool PHSP){
gStyle -> SetOptStat(0);
gROOT->SetBatch(kTRUE);
TH1::SetDefaultSumw2(kTRUE);
//Load all files
coutInfo("Opennig the trees");
TChain *tree= new TChain("DecayTreeTruthMatched");
//Open the file(s)
if (Run==0){
tree->Add(returnFileAddress(year, getRunID(year), magnet, true, false, true, ReferenceChannel, PHSP, false).c_str());
coutDebug("Adding " + returnFileAddress(year, getRunID(year), magnet, true, false, true, ReferenceChannel, PHSP, false));
}
else{
for (auto yr: yearsMC(ReferenceChannel, PHSP, Run)){
tree->Add(returnFileAddress(yr, getRunID(yr), magnet, true, false, true, ReferenceChannel, PHSP, false).c_str());
coutDebug("Adding " + returnFileAddress(yr, getRunID(yr), magnet, true, false, true, ReferenceChannel, PHSP, false));
}
}
//activate all branches
tree->SetBranchStatus("*",1);
//Draw the histogram
TH1D *h_tmp = new TH1D ("h_tmp", "h_tmp", nBins,lowEdge,highEdge);
tree->Draw(Form("%s>>h_tmp",var.c_str()));
//Get the binnings
getSimilarlyPopulatedBins(h_tmp,desiredBins, nBins);
//free da memory
delete h_tmp;
delete tree;
return;
}
/////////////////////////////////////////////
//
// Get result from a fit file
// Comment out the content of the functions in order to compile getPathForPython *facepalm
//
/////////////////////////////////////////////
RooFitResult* getResult(TFile *fitFile){
RooFitResult* fitResult;
fitFile->GetObject("fitresult_pdf_data",fitResult);
if (verboseLevel < 2) fitResult->Print();
return fitResult;
}
RooRealVar* getVarFromResult(RooFitResult *fitResult, string name){
RooRealVar *myVar = (RooRealVar*)fitResult->floatParsFinal().find(name.c_str());
return myVar;
}
double getBplusMeanFromResult(TFile* fitFile){
RooRealVar *mass = getVarFromResult(getResult(fitFile),"sig_mean");
return mass->getVal();
}
//Read the yields from a fit file
double getSigYield(TFile *fitFile){
TVectorD *yield = (TVectorD*)fitFile->Get("yield");
return (*yield)[0];
}
double getSigYieldErr(TFile *fitFile){
TVectorD *yield_err = (TVectorD*)fitFile->Get("yield_err");
return (*yield_err)[0];
}
double getBkgYield(TFile *fitFile){
TVectorD *bkg = (TVectorD*)fitFile->Get("background");
return (*bkg)[0];
}
double getBkgYieldErr(TFile *fitFile){
TVectorD *bkg_eff = (TVectorD*)fitFile->Get("background_err");
return (*bkg_eff)[0];
}
double getEffSigma(TFile *fitFile){
TVectorD *sigma = (TVectorD*)fitFile->Get("sigma_eff");
return (*sigma)[0];
}
/////////////////////////////////////////////
//
// Histogram/graph helpers
//
/////////////////////////////////////////////
TH1D *convertTGraph(TGraph *graph){
//Possibly fix tihs at some point to be generally repflecting TMefficiencyClass
double binEdges[7] = {0.1e6, 4.0e6, 8.0e6, 11.0e6, 12.5e6, 15.0e6, 20.0e6};
TH1D *hist = new TH1D("hist","hist",6,binEdges);
auto nPoints = graph->GetN(); // number of points in your TGraph
for(int i=0; i < nPoints; ++i) {
double x,y;
graph->GetPoint(i, x, y);
hist->Fill(x,y); // ?
}
for (int b=0; b < hist->GetXaxis()->GetNbins(); b++){
hist->SetBinError(b+1,0);
}
return hist;
}
/////////////////////////////////////////////
//
// Get numbers of MC events
//
/////////////////////////////////////////////
double generated_basic_events = 10000.0;
double generated_basic_events_PHSP = 1000.0;
double get_generated_events(bool PHSP){
if (PHSP) return generated_basic_events_PHSP;
else return generated_basic_events;
}
vector<int> generated_signal_events_down = {507551, 514015, 0, 1000281, 1151738, 1235528};
vector<int> generated_signal_events_up = {502787, 500458, 0, 1000281, 1153816, 1196153};
vector<int> generated_reference_events_down = {1011831, 1003888, 1007712, 1010609, 0, 0};
vector<int> generated_reference_events_up = {1007920, 1000278, 1009484, 1000281, 0, 0};
vector<int> generated_PHSP_events_down = {85736, 226933, 95323, 264917, 246457, 291850};
vector<int> generated_PHSP_events_up = {86004, 226430, 94980, 267674, 242673, 299252};
vector<double> gen_tables_signal_efficiency_down = {0.14388, 0.14769, 0.1615, 0.1610, 0.1609, 0.1605};//RUN I taken from Reference Channel
vector<double> gen_tables_signal_efficiency_up = {0.14422, 0.14787, 0.1608, 0.1611, 0.1595, 0.1609};//RUN I taken from Reference Channel
vector<double> gen_tables_reference_efficiency_down = {0.14388, 0.14769, 0.1581, 0.1585, 0.1585, 0.1585}; //2017+2018 taken as 2016
vector<double> gen_tables_reference_efficiency_up = {0.14422, 0.14787, 0.1574, 0.1590, 0.1590, 0.1590}; //2017+2018 taken as 2016
vector<int> yield_signal_events = {5451, 4906, 4096, 10382, 15994, 14919};
vector<int> yield_signal_events_err = {70, 64, 1, 96, 15, 116};
vector<int> yield_reference_events = {19730, 17579, 18392, 23965, 0, 0};
vector<int> yield_reference_events_err = {135, 127, 130, 149, 0, 0};
vector<int> yield_PHSP_events = {14091, 33086, 13530, 48135, 54030, 61641};
vector<int> yield_PHSP_events_err = {113, 174, 111, 209, 221, 237};
vector<int> yield_signal_events_notTM = {7056, 6587, 6354, 15652, 20207, 19734};
vector<int> yield_signal_events_notTM_err = {456, 61, 81, 55, 368, 222};
vector<int> yield_reference_events_notTM = {24375, 22698, 23403, 30303, 0, 0};
vector<int> yield_reference_events_notTM_err = {236, 123, 143, 12, 0, 0};
vector<int> yield_signal_events_notTM_unique = {5854, 5304, 5155, 13086, 17055, 16374};
vector<int> yield_signal_events_notTM_unique_err = {89, 106, 86, 169, 172, 211};
vector<int> yield_reference_events_notTM_unique = {20564, 18772, 19204, 25140, 0, 0};
vector<int> yield_reference_events_notTM_unique_err = {210, 123, 89, 167, 0, 0};
int get_position_from_year(string year){
int position = year.back() - '0'; //get last char of a string and convert it to an int
return position>4 ? position-3 : position-1; //11->0, 12->1, 15->2, 16->3, ...
}
int get_gen_evts(string year, bool ReferenceChannel, bool PHSP){
int pos = get_position_from_year(year);
if (ReferenceChannel) return generated_reference_events_down.at(pos) + generated_reference_events_up.at(pos);
if (PHSP) return generated_PHSP_events_down.at(pos) + generated_PHSP_events_up.at(pos);
return generated_signal_events_down.at(pos) + generated_signal_events_up.at(pos);
}
int get_gen_evts(int Run, bool ReferenceChannel, bool PHSP){
int yield = 0;
for (auto yr: yearsMC(ReferenceChannel,PHSP,Run)) yield += get_gen_evts(yr,ReferenceChannel,PHSP);
return yield;
}
int get_selected_evts(string year, bool ReferenceChannel, bool PHSP){
int pos = get_position_from_year(year);
if (ReferenceChannel) return yield_reference_events.at(pos);
if (PHSP) return yield_PHSP_events.at(pos);
return yield_signal_events.at(pos);
}
int get_selected_evts(int Run, bool ReferenceChannel, bool PHSP){
int yield = 0;
for (auto yr: yearsMC(ReferenceChannel,PHSP,Run)) yield += get_selected_evts(yr,ReferenceChannel,PHSP);
return yield;
}
int get_selected_evts_err(string year, bool ReferenceChannel, bool PHSP){
int pos = get_position_from_year(year);
if (ReferenceChannel) return yield_reference_events_err.at(pos);
if (PHSP) return yield_PHSP_events_err.at(pos);
return yield_signal_events_err.at(pos);
}
double get_selection_efficiency(string year, bool ReferenceChannel, bool PHSP){
return double(get_selected_evts(year,ReferenceChannel,PHSP))/double(get_gen_evts(year,ReferenceChannel,PHSP));
}
double get_selection_efficiency(int Run, bool ReferenceChannel, bool PHSP){
return double(get_selected_evts(Run,ReferenceChannel,PHSP))/double(get_gen_evts(Run,ReferenceChannel,PHSP));
}
double get_tables_eff(string year, bool ReferenceChannel){ //Get efficiency in a year as an average between up and down
int pos = get_position_from_year(year);
if (ReferenceChannel) return (gen_tables_reference_efficiency_down.at(pos)+gen_tables_reference_efficiency_up.at(pos))/2;
else return (gen_tables_signal_efficiency_down.at(pos)+gen_tables_signal_efficiency_up.at(pos))/2;
}
double get_tables_eff(int Run, bool ReferenceChannel){ //Get efficiency of a run as an average between years
double sum = 0;
int nEntries = 0;
double tmp_eff = 0;
for (auto yr: yearsMC(ReferenceChannel,false,Run)){
tmp_eff = get_tables_eff(yr, ReferenceChannel);
if (tmp_eff!=0) nEntries++;
sum += tmp_eff;
}
coutDebug("Total eff/nEntries: " + to_string(sum) + "/" + to_string(nEntries) );
if (nEntries == 0) return 0;
return sum/nEntries;
}
void print_all_yields_and_efficiencies(bool ReferenceChannel, bool PHSP){
for (auto yr : yearsMC(ReferenceChannel,PHSP, 12)){
cout << "------ year " << yr << " ------" << endl;
cout << "\t\t Generated events:\t" << get_gen_evts(yr, ReferenceChannel, PHSP) << endl;
cout << "\t\t Selected events:\t" << get_selected_evts(yr, ReferenceChannel, PHSP) << endl;
cout << "\t\t Efficiency: \t" << get_selection_efficiency(yr, ReferenceChannel, PHSP) << endl;
}
cout << "------ Run 1 ------" << endl;
cout << "\t\t Generated events:\t" << get_gen_evts(1, ReferenceChannel, PHSP) << endl;
cout << "\t\t Selected events:\t" << get_selected_evts(1, ReferenceChannel, PHSP) << endl;
cout << "\t\t Efficiency: \t" << get_selection_efficiency(1, ReferenceChannel, PHSP) << endl;
cout << "------ Run 2 ------" << endl;
cout << "\t\t Generated events:\t" << get_gen_evts(2, ReferenceChannel, PHSP) << endl;
cout << "\t\t Selected events:\t" << get_selected_evts(2, ReferenceChannel, PHSP) << endl;
cout << "\t\t Efficiency: \t" << get_selection_efficiency(2, ReferenceChannel, PHSP) << endl;
return;
}
//----------------------------------------------------------------------------------------------------------
#endif // UTILS_HPP