forked from osWW-VBS/nanoAOD_vvVBS
-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathJetSFMaker.py
123 lines (99 loc) · 5.44 KB
/
JetSFMaker.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
# from https://github.com/latinos/LatinoAnalysis/blob/master/NanoGardener/python/modules/JetSFMaker.py
import os
import ROOT
from PhysicsTools.NanoAODTools.postprocessing.framework.eventloop import Module
from PhysicsTools.NanoAODTools.postprocessing.framework.datamodel import Collection
class JetSFMaker(Module):
#----------------------------------------------------------------------------
#Add branches for Jet PUID scale factors and up/down SF variations (per jet).
#Per event SF should be calculated as a product of per jet SFs and applied as
#weight. Same for up/down variations (weights).
#----------------------------------------------------------------------------
def __init__(self, cmssw, puid_sf_config='PhysicsTools/NanoAODTools/python/postprocessing/analysis/nanoAOD_skim/data/JetPUID_cfg.py'):
cmssw_base = os.getenv('CMSSW_BASE')
with open(cmssw_base + '/src/' + puid_sf_config) as src:
exec(src)
puid_sf_cfg = jet_puid_sf[cmssw]
source = ROOT.TFile.Open(cmssw_base + '/src/' + puid_sf_cfg['source'])
self.sf_maps = {}
self.sf_uncty_maps = {}
self.eff_maps = {}
for jtype in ['real', 'pu']:
for wp, iwp in [('loose', 'L'), ('medium', 'M'), ('tight', 'T')]:
key = '%s_%s' % (jtype, wp)
key_uncty = '%s_%s_uncty' % (jtype, wp)
key_eff = '%s_mc_%s' % (jtype, wp)
self.sf_maps[key] = source.Get(puid_sf_cfg[key])
self.sf_maps[key].SetDirectory(0)
self.sf_uncty_maps[key_uncty] = source.Get(puid_sf_cfg[key_uncty])
self.sf_uncty_maps[key_uncty].SetDirectory(0)
self.eff_maps[key_eff] = source.Get(puid_sf_cfg[key_eff])
self.eff_maps[key_eff].SetDirectory(0)
source.Close()
def beginFile(self, inputFile, outputFile, inputTree, wrappedOutputTree):
self.out = wrappedOutputTree
for wp in ['loose', 'medium', 'tight']:
self.out.branch('Jet_PUIDSF_%s' % wp, 'F', lenVar='nJet')
self.out.branch('Jet_PUIDSF_%s_up' % wp, 'F', lenVar='nJet')
self.out.branch('Jet_PUIDSF_%s_down' % wp, 'F', lenVar='nJet')
def analyze(self, event):
jets = Collection(event, 'Jet')
sfs = {'loose': [], 'medium': [], 'tight': []}
sfs_up = {'loose': [], 'medium': [], 'tight': []}
sfs_down = {'loose': [], 'medium': [], 'tight': []}
for jet in jets:
if jet.genJetIdx == -1:
jtype = 'pu'
else:
jtype = 'real'
for iwp,wp in [(2,'loose'), (1,'medium'), (0,'tight')]:
#get ingredients for calculating per jet weights
sf, stat_err, syst_err, eff = self.get_sf_and_eff(jtype, wp, jet)
passed_puid = bool(jet.puId & (1 << iwp))
#calculate per jet weight + up/down per jet weight variations
puid_jw = 1.
puid_upjw = 1.
puid_downjw = 1.
if passed_puid:
puid_jw = sf
if (jtype == 'real') or (jtype == 'pu' and abs(jet.eta)>=2.5):
up = sf + syst_err + stat_err
down = sf - syst_err - stat_err
else:
up = 1 + abs(sf-1)
down = 1 - abs(sf-1)
puid_upjw = up
puid_downjw = down
else:
puid_jw = (1.-sf*eff)/(1.-eff)
if (jtype == 'real') or (jtype == 'pu' and abs(jet.eta)>=2.5):
up = sf + syst_err + stat_err
down = sf - syst_err - stat_err
else:
up = 1 + abs(sf-1)
down = 1 - abs(sf-1)
puid_upjw = (1.-up*eff)/(1.-eff)
puid_downjw = (1.-down*eff)/(1.-eff)
#store per jet weights and variations
sfs[wp].append(puid_jw)
sfs_up[wp].append(puid_upjw)
sfs_down[wp].append(puid_downjw)
for wp in ['loose', 'medium', 'tight']:
self.out.fillBranch('Jet_PUIDSF_%s' % wp, sfs[wp])
self.out.fillBranch('Jet_PUIDSF_%s_up' % wp, sfs_up[wp])
self.out.fillBranch('Jet_PUIDSF_%s_down' % wp, sfs_down[wp])
return True
def get_sf_and_eff(self, jtype, wp, jet):
sf_map = self.sf_maps['%s_%s' % (jtype, wp)]
sf_uncty_map = self.sf_uncty_maps['%s_%s_uncty' % (jtype, wp)]
eff_map = self.eff_maps['%s_mc_%s' % (jtype, wp)]
if jet.pt < 30. or jet.pt > 50. or abs(jet.eta) > 4.7:
#do not apply SF outside CleanJet region and where PUID was not applied
return 1.,0.,0.,0.
ix = min(max(1, sf_map.GetXaxis().FindFixBin(jet.pt)), sf_map.GetNbinsX())
iy = min(max(1, sf_map.GetYaxis().FindFixBin(jet.eta)), sf_map.GetNbinsY())
jx = min(max(1, sf_uncty_map.GetXaxis().FindFixBin(jet.pt)), sf_uncty_map.GetNbinsX())
jy = min(max(1, sf_uncty_map.GetYaxis().FindFixBin(jet.eta)), sf_uncty_map.GetNbinsY())
kx = min(max(1, eff_map.GetXaxis().FindFixBin(jet.pt)), eff_map.GetNbinsX())
ky = min(max(1, eff_map.GetYaxis().FindFixBin(jet.eta)), eff_map.GetNbinsY())
return sf_map.GetBinContent(ix, iy), sf_map.GetBinError(ix, iy), sf_uncty_map.GetBinContent(jx, jy), eff_map.GetBinContent(kx, ky)