ATLAS Offline Software
Loading...
Searching...
No Matches
python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock Class Reference
Inheritance diagram for python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock:
Collaboration diagram for python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock:

Public Member Functions

 __init__ (self)
None makeTriggerGlobalEffCorrAlg (self, ConfigAccumulator config, matchingTool, bool noSF, str triggerSuffix='')
None createTrigMatching (self, ConfigAccumulator config, matchingTool, particles, str trig_string="", set trig_chains=None, set trig_chains_dummy=None)
None makeTrigMatchingAlg (self, ConfigAccumulator config, matchingTool)
None makeAlgs (self, ConfigAccumulator config)

Public Attributes

 separateChainMatching
 triggerChainsPerYear
 electrons
 muons
 photons
 postfix
 triggerMatchingChainsPerYear
 includeAllYearsPerRun
 taus
dict multiTriggerChainsPerYear
 noGlobalTriggerEff
 noEffSF

Detailed Description

the ConfigBlock for trigger analysis

Definition at line 77 of file TriggerAnalysisSFConfig.py.

Constructor & Destructor Documentation

◆ __init__()

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.__init__ ( self)

Definition at line 79 of file TriggerAnalysisSFConfig.py.

79 def __init__(self):
80 super(TriggerAnalysisSFBlock, self).__init__()
81 self.addDependency('Electrons', required=False)
82 self.addDependency('Photons', required=False)
83 self.addDependency('Muons', required=False)
84 self.addDependency('Taus', required=False)
85 self.addDependency('OverlapRemoval', required=False)
86
87 self.addOption ('triggerChainsPerYear', {}, type=None,
88 info="a dictionary with key (string) the year and value (list of "
89 "strings) the trigger chains. You can also use `||` within a string "
90 "to enforce an OR of triggers without looking up the individual "
91 "triggers. Used for both trigger selection and SFs.")
92 self.addOption ('multiTriggerChainsPerYear', {}, type=None,
93 info="a dictionary with key (string) a trigger set name and value a "
94 "`triggerChainsPerYear` dictionary, following the previous convention. "
95 "Relevant for analyses using different triggers in different categories, "
96 "where the trigger global scale factors shouldn't be combined.")
97 self.addOption ('noFilter', False, type=bool,
98 info="do not apply an event filter (i.e. keep events "
99 "not passing trigger selection and matching).")
100 self.addOption ('electronID', '', type=str,
101 info="the electron ID WP to use.")
102 self.addOption ('electronIsol', '', type=str,
103 info="the electron isolation WP to use.")
104 self.addOption ('photonIsol', '', type=str,
105 info="the photon isolation WP to use.")
106 self.addOption ('muonID', '', type=str,
107 info="the muon quality WP to use.")
108 self.addOption ('electrons', '', type=str,
109 info="the input electron container, with a possible selection, in "
110 "the format `container` or `container.selection`.")
111 self.addOption ('muons', '', type=str,
112 info="the input muon container, with a possible selection, in the "
113 "format `container` or `container.selection`.")
114 self.addOption ('photons', '', type=str,
115 info="the input photon container, with a possible selection, in "
116 "the format `container` or `container.selection`.")
117 self.addOption ('taus', '', type=str,
118 info="the input tau-jet container, with a possible selection, in "
119 "the format `container` or `container.selection`.")
120 self.addOption ('numberOfToys', 0, type=int,
121 info="the number of toy experiments to run to estimate the trigger efficiencies, "
122 "instead of using explicit formulas. Set it to 0 to not use toys.")
123 self.addOption ('noEffSF', False, type=bool,
124 info="disables the calculation of efficiencies and scale factors. "
125 "Experimental! only useful to test a new WP for which scale "
126 "factors are not available. Still performs the global trigger "
127 "matching (same behaviour as on data).",
128 expertMode=True)
129 self.addOption ('noGlobalTriggerEff', False, type=bool,
130 info="disables the global trigger efficiency tool (including "
131 "matching), which is only suited for electron/muon/photon "
132 "trigger legs.")
133 self.addOption ('separateChainMatching', False, type=bool,
134 info="store the matching status for each trigger separately.")
135 self.addOption ('triggerMatchingChainsPerYear', {}, type=None,
136 info="a dictionary with key (string) the year and value (list of "
137 "strings) the trigger chains.")
138 self.addOption("includeAllYearsPerRun", False, type=bool,
139 info="trigger matching will include all configured years "
140 "in the LHC run in all jobs.")
141 self.addOption ('postfix', '', type=str,
142 info="a unique identifier for the trigger matching decorations. Only "
143 "useful when defining multiple setups.")
144

Member Function Documentation

◆ createTrigMatching()

None python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.createTrigMatching ( self,
ConfigAccumulator config,
matchingTool,
particles,
str trig_string = "",
set trig_chains = None,
set trig_chains_dummy = None )

Definition at line 224 of file TriggerAnalysisSFConfig.py.

232 ) -> None:
233 if not particles or (not trig_chains and not trig_chains_dummy):
234 return
235
236 if not any(trig_string in trig for trig in trig_chains) \
237 and not any(trig_string in trig for trig in trig_chains_dummy):
238 return
239
240 alg = config.createAlgorithm("CP::TrigMatchingAlg", f"TrigMatchingAlg_{trig_string}{self.postfix}")
241 alg.matchingTool = f"{matchingTool.getType()}/{matchingTool.getName()}"
242 alg.matchingDecoration = f"trigMatched{self.postfix}"
243 alg.trigSingleMatchingList = [trig for trig in trig_chains if trig_string in trig]
244 alg.trigSingleMatchingListDummy = [trig for trig in trig_chains_dummy if trig_string in trig]
245 alg.particles, alg.particleSelection = config.readNameAndSelection(particles)
246
247 for trig in list(alg.trigSingleMatchingList) + list(alg.trigSingleMatchingListDummy):
248 trig = trig.replace(".", "p").replace("-", "_").replace(" ", "")
249 if trig_string in trig:
250 config.addOutputVar(particles.split(".")[0], f"trigMatched_{self.postfix}{trig}", f"trigMatched_{self.postfix}{trig}", noSys=True)
251
std::string replace(std::string s, const std::string &s2, const std::string &s3)
Definition hcg.cxx:310

◆ makeAlgs()

None python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.makeAlgs ( self,
ConfigAccumulator config )

Definition at line 282 of file TriggerAnalysisSFConfig.py.

282 def makeAlgs(self, config: ConfigAccumulator) -> None:
283 if (
284 self.multiTriggerChainsPerYear and
285 self.triggerChainsPerYear and
286 self.triggerChainsPerYear not in self.multiTriggerChainsPerYear.values()
287 ):
288 raise Exception('multiTriggerChainsPerYear and triggerChainsPerYear cannot be configured at the same time!')
289
290 if self.triggerChainsPerYear and not self.multiTriggerChainsPerYear:
291 self.multiTriggerChainsPerYear = {'': self.triggerChainsPerYear}
292
293 # Create the decision algorithm, keeping track of the decision tool for later
294 decisionTool = TriggerAnalysisBlock.makeTriggerDecisionTool(config)
295
296 # Now pass it to the matching algorithm, keeping track of the matching tool for later
297 matchingTool = TriggerAnalysisBlock.makeTriggerMatchingTool(config, decisionTool)
298
299 # Calculate multi-lepton (electron/muon/photon) trigger efficiencies and SFs
300 if self.multiTriggerChainsPerYear and not self.noGlobalTriggerEff:
301 for suffix, trigger_chains in self.multiTriggerChainsPerYear.items():
302 self.triggerChainsPerYear = trigger_chains
303 self.makeTriggerGlobalEffCorrAlg(config, matchingTool, self.noEffSF, suffix)
304
305 # Save trigger matching information (currently only single leg trigger are supported)
306 if self.triggerMatchingChainsPerYear:
307 self.makeTrigMatchingAlg(config, matchingTool)
308
309 return

◆ makeTriggerGlobalEffCorrAlg()

None python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.makeTriggerGlobalEffCorrAlg ( self,
ConfigAccumulator config,
matchingTool,
bool noSF,
str triggerSuffix = '' )

Definition at line 145 of file TriggerAnalysisSFConfig.py.

151 ) -> None:
152 alg = config.createAlgorithm( 'CP::TrigGlobalEfficiencyAlg', 'TrigGlobalSFAlg' + triggerSuffix + self.postfix)
153 if config.geometry() is LHCPeriod.Run3:
154 alg.triggers_2022 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2022)]
155 alg.triggers_2023 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2023)]
156 alg.triggers_2024 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2024)]
157 alg.triggers_2025 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2025)]
158 if is_mc_from(config, Campaign.MC23a) or is_data_from(config, 2022):
159 if not alg.triggers_2022:
160 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the year 2022!')
161 elif is_mc_from(config, Campaign.MC23d) or is_data_from(config, 2023):
162 if not alg.triggers_2023:
163 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the year 2023!')
164 else:
165 alg.triggers_2015 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2015)]
166 alg.triggers_2016 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2016)]
167 alg.triggers_2017 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2017)]
168 alg.triggers_2018 = [trig.replace("HLT_","").replace(" || ", "_OR_") for trig in get_year_data(self.triggerChainsPerYear, 2018)]
169 if is_mc_from(config, Campaign.MC20a):
170 if not (alg.triggers_2015 and alg.triggers_2016):
171 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the years 2015 and 2016!')
172 elif is_data_from(config, 2015):
173 if not alg.triggers_2015:
174 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the year 2015!')
175 elif is_data_from(config, 2016):
176 if not alg.triggers_2016:
177 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the year 2016!')
178 elif is_mc_from(config, Campaign.MC20d) or is_data_from(config, 2017):
179 if not alg.triggers_2017:
180 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the year 2017!')
181 elif is_mc_from(config, Campaign.MC20e) or is_data_from(config, 2018):
182 if not alg.triggers_2018:
183 raise ValueError('TriggerAnalysisConfig: you must provide a set of triggers for the year 2018!')
184
185 triggerMatchingChains = set()
186 if self.separateChainMatching:
187 for year in get_input_years(config):
188 for trig in get_year_data(self.triggerChainsPerYear, year):
189 triggerMatchingChains.update(trig.replace(' || ', '_OR_').split('_OR_'))
190 alg.separateMatchingTriggers = list(triggerMatchingChains)
191 alg.separateMatchingDecorationSuffix = triggerSuffix + self.postfix
192
193 alg.matchingTool = '%s/%s' % ( matchingTool.getType(), matchingTool.getName() )
194 alg.isRun3Geo = config.geometry() is LHCPeriod.Run3
195 alg.numberOfToys = self.numberOfToys
196 alg.scaleFactorDecoration = 'globalTriggerEffSF' + triggerSuffix + self.postfix + '_%SYS%'
197 alg.matchingDecoration = 'globalTriggerMatch' + triggerSuffix + self.postfix + '_%SYS%'
198 alg.eventDecisionOutputDecoration = 'globalTriggerMatch' + triggerSuffix + self.postfix + '_dontsave_%SYS%'
199 alg.doMatchingOnly = config.dataType() is DataType.Data or noSF
200 alg.noFilter = self.noFilter
201 alg.electronID = self.electronID
202 alg.electronIsol = self.electronIsol
203 alg.photonIsol = self.photonIsol
204 alg.muonID = self.muonID
205 if self.electrons:
206 alg.electrons, alg.electronSelection = config.readNameAndSelection(self.electrons)
207 if self.muons:
208 alg.muons, alg.muonSelection = config.readNameAndSelection(self.muons)
209 if self.photons:
210 alg.photons, alg.photonSelection = config.readNameAndSelection(self.photons)
211 if not (self.electrons or self.muons or self.photons):
212 raise ValueError('TriggerAnalysisConfig: at least one object collection must be provided! (electrons, muons, photons)' )
213
214 if config.dataType() is not DataType.Data and not alg.doMatchingOnly:
215 config.addOutputVar('EventInfo', alg.scaleFactorDecoration, 'globalTriggerEffSF' + triggerSuffix + self.postfix)
216 config.addOutputVar('EventInfo', alg.matchingDecoration, 'globalTriggerMatch' + triggerSuffix + self.postfix, noSys=False)
217
218 for trig in triggerMatchingChains:
219 var = f'triggerMatch_{trig.replace("-", "_").replace(".", "p")}{alg.separateMatchingDecorationSuffix}'
220 config.addOutputVar('EventInfo', f'{var}_%SYS%', var, noSys=False)
221
222 return
223
STL class.
std::vector< std::string > split(const std::string &s, const std::string &t=":")
Definition hcg.cxx:177

◆ makeTrigMatchingAlg()

None python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.makeTrigMatchingAlg ( self,
ConfigAccumulator config,
matchingTool )

Definition at line 252 of file TriggerAnalysisSFConfig.py.

256 ) -> None:
257 years = get_input_years(config)
258
259 triggerMatchingChains = set()
260 triggerMatchingChainsDummy = set()
261 for year in years:
262 for trig in get_year_data(self.triggerMatchingChainsPerYear, year):
263 trig = trig.replace(' || ', '_OR_')
264 triggerMatchingChains.update(trig.split('_OR_'))
265 if self.includeAllYearsPerRun:
266 triggerMatchingChainsAll = set()
267 for year in self.triggerMatchingChainsPerYear:
268 if not is_year_in_current_period(config, year):
269 continue
270 for trig in get_year_data(self.triggerMatchingChainsPerYear, year):
271 trig = trig.replace(' || ', '_OR_')
272 triggerMatchingChainsAll.update(trig.split('_OR_'))
273 triggerMatchingChainsDummy = triggerMatchingChainsAll - triggerMatchingChains
274
275 self.createTrigMatching(config, matchingTool, self.electrons, 'HLT_e', triggerMatchingChains, triggerMatchingChainsDummy)
276 self.createTrigMatching(config, matchingTool, self.muons, 'HLT_mu', triggerMatchingChains, triggerMatchingChainsDummy)
277 self.createTrigMatching(config, matchingTool, self.photons, 'HLT_g', triggerMatchingChains, triggerMatchingChainsDummy)
278 self.createTrigMatching(config, matchingTool, self.taus, 'HLT_tau', triggerMatchingChains, triggerMatchingChainsDummy)
279
280 return
281

Member Data Documentation

◆ electrons

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.electrons

Definition at line 205 of file TriggerAnalysisSFConfig.py.

◆ includeAllYearsPerRun

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.includeAllYearsPerRun

Definition at line 265 of file TriggerAnalysisSFConfig.py.

◆ multiTriggerChainsPerYear

dict python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.multiTriggerChainsPerYear

Definition at line 290 of file TriggerAnalysisSFConfig.py.

◆ muons

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.muons

Definition at line 207 of file TriggerAnalysisSFConfig.py.

◆ noEffSF

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.noEffSF

Definition at line 303 of file TriggerAnalysisSFConfig.py.

◆ noGlobalTriggerEff

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.noGlobalTriggerEff

Definition at line 300 of file TriggerAnalysisSFConfig.py.

◆ photons

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.photons

Definition at line 209 of file TriggerAnalysisSFConfig.py.

◆ postfix

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.postfix

Definition at line 215 of file TriggerAnalysisSFConfig.py.

◆ separateChainMatching

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.separateChainMatching

Definition at line 186 of file TriggerAnalysisSFConfig.py.

◆ taus

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.taus

Definition at line 278 of file TriggerAnalysisSFConfig.py.

◆ triggerChainsPerYear

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.triggerChainsPerYear

Definition at line 188 of file TriggerAnalysisSFConfig.py.

◆ triggerMatchingChainsPerYear

python.TriggerAnalysisSFConfig.TriggerAnalysisSFBlock.triggerMatchingChainsPerYear

Definition at line 262 of file TriggerAnalysisSFConfig.py.


The documentation for this class was generated from the following file: