4 JetRecConfig: A helper module for configuring jet reconstruction
6 The functions defined here turn JetDefinition object into ComponentAccumulator or list of algs fully configured
7 and ready to be inserted in the framework sequence.
9 Author: TJ Khoo, P-A Delsart
13 from AthenaCommon
import Logging
14 jetlog = Logging.logging.getLogger(
'JetRecConfig')
16 from ROOT
import xAODType
19 from AthenaConfiguration.ComponentAccumulator
import ComponentAccumulator
20 from AthenaConfiguration.ComponentFactory
import CompFactory
23 from JetRecConfig.JetDefinition
import JetDefinition, JetInputConstitSeq, JetInputConstit, JetInputExternal
24 from JetRecConfig.JetGrooming
import GroomingDefinition
25 from JetRecConfig.DependencyHelper
import solveDependencies, solveGroomingDependencies, aliasToModDef
28 __all__ = [
"JetRecCfg",
"JetInputCfg"]
36 def JetRecCfg( flags, jetdef, returnConfiguredDef=False):
37 """Top-level function for running jet finding or grooming.
39 This returns a ComponentAccumulator that can be merged with others
40 from elsewhere in the job and which provides everything needed to
41 reconstruct one jet collection.
44 - jetdef : jet or grooming definition
45 - flags : the configuration flags instance, mainly for input file
46 peeking such that we don't attempt to reproduce stuff that's already
47 in the input file. And also to be able to invoke building of inputs outside of Jet domain during reco from RAW/RDO.
48 - returnConfiguredDef : is for debugging. It will also returns the cloned JetDefinition which contains the calculated dependencies.
52 sequenceName = jetdef.fullname()
53 jetlog.info(
"******************")
54 jetlog.info(
"Setting up to find {0}".
format(sequenceName))
57 from AthenaCommon.CFElements
import parOR
58 components.addSequence(
parOR(sequenceName) )
61 if isinstance(jetdef, JetDefinition):
63 elif isinstance(jetdef, GroomingDefinition):
67 if flags.Concurrency.NumThreads <= 0:
68 jetlog.info(
"Reordering algorithms in sequence {0}".
format(sequenceName))
74 if isinstance(a, ComponentAccumulator):
77 components.addEventAlgo( a , sequenceName = sequenceName )
79 if returnConfiguredDef:
return components, jetdef_i
84 """Returns a ComponentAccumulator containing algs needed to build inputs to jet finding as defined by jetOrConstitdef
86 jetOrConstitdef can either be
87 * a JetDefinition : this happens when called from JetRecCfg, then the jetdef._prereqDic/Order are used.
88 * a JetInputConstit : to allow scheduling the corresponding constituents algs independently of any jet alg.
90 context is only used if jetOrConstitdef is not a JetDefinition and must refer to a context in StandardJetContext
98 if isinstance(a, ComponentAccumulator):
101 components.addEventAlgo(a)
106 """Builds a ComponentAccumulator for creating PseudoJetContainer needed by jetdef.
107 THIS updates jetdef._internalAtt['finalPJContainer']
111 for pjalg
in pjalglist:
112 components.addEventAlgo(pjalg)
123 """ Create the algorithms necessary to build the jet collection defined by jetdef.
125 This internally finds all the dependencies declared into jetdef (through input, ghosts & modifiers)
126 and returns a list of all necessary algs.
128 if returnConfiguredDef==True, also returns the fully configured clone of jetdef containing solved dependencies (debugging)
130 monTool is to allow the trigger config to pass a monitoring tool.
132 returns a list containing either algs or ComponentAccumulator
133 (ComponentAccumulator occurs only (?) in reco from RDO/RAW when we need to build externals such as clusters or tracks : in this case we call the main config functions from external packages)
147 if returnConfiguredDef:
164 jetlog.info(
"Scheduled JetAlgorithm instance \"jetalg_{0}\"".
format(jetdef_i.fullname()))
166 if returnConfiguredDef:
167 return algs, jetdef_i
171 """Instantiate and schedule all the algorithms needed to run the grooming alg 'groomdef' and
172 add them in the ComponentAccumulator 'components'
174 This function is meant to be called from the top-level JetRecConfig.JetRecCfg
175 (groomdef is expected to be non locked and will be modified).
177 monTool is to allow the trigger config to pass a monitoring tool.
186 for prereq
in groomdef_i._prereqOrder:
188 if len(prereq.split(
':')) > 2:
190 reqType, reqKey = prereq.split(
':')
192 groomdef_i.ungroomeddef.ghostdefs.append(reqKey)
193 elif reqType.endswith(
'input') :
194 groomdef_i.ungroomeddef.extrainputs.append(reqKey)
196 jetlog.info(
"Scheduling parent alg {} for {} ".
format(groomdef.ungroomeddef.fullname(), groomdef.fullname()))
201 algs, ungroomeddef_i =
getJetDefAlgs(flags, groomdef_i.ungroomeddef ,
True)
202 groomdef_i._ungroomeddef = ungroomeddef_i
210 jetlog.info(
"Scheduled JetAlgorithm instance \"jetalg_{0}\"".
format(groomdef_i.fullname()))
212 if returnConfiguredDef:
return algs, groomdef_i
216 def getJetAlgs(flags, jetdef, returnConfiguredDef=False, monTool=None):
218 if isinstance(jetdef, JetDefinition):
220 elif isinstance(jetdef, GroomingDefinition):
221 func = getJetGroomAlgs
223 return func(flags, jetdef, returnConfiguredDef, monTool)
234 """ Builds the list of configured PseudoJetAlgorithm needed for this jetdef.
235 THIS updates jetdef._internalAtt['finalPJContainer']
236 (this function is factorized out of PseudoJetCfg so it can be used standalone in the trigger config)
239 constitpjalg =
getConstitPJGAlg(jetdef.inputdef , suffix=
None , flags=jetdef._cflags, parent_jetdef = jetdef)
241 finalPJContainer =
str(constitpjalg.OutputContainer)
242 pjalglist = [constitpjalg]
245 ghostlist = [ key
for key
in jetdef._prereqOrder
if key.startswith(
'ghost:')]
248 pjContNames = [finalPJContainer]
249 for ghostkey
in sorted(ghostlist):
250 ghostdef = jetdef._prereqDic[ghostkey]
252 pjalglist.append(ghostpjalg)
253 pjContNames.append(
str(ghostpjalg.OutputContainer) )
257 finalPJContainer =
str(finalPJContainer)+
"_merged"+mergeId
258 mergerName =
"PJMerger_id"+mergeId
259 mergeAlg =CompFactory.PseudoJetMerger(
261 InputPJContainers = pjContNames,
262 OutputContainer = finalPJContainer,
264 pjalglist.append(mergeAlg)
267 jetdef._internalAtt[
'finalPJContainer'] = finalPJContainer
271 _mergedPJContainers = dict()
273 """returns a simple unique ID for the list of PseudoJet container in pjList"""
274 t = tuple(
str(n)
for n
in pjList)
275 currentSize = len(_mergedPJContainers)
276 return str(_mergedPJContainers.setdefault(t, currentSize))
279 def getInputAlgs(jetOrConstitdef, flags, context="default", monTool=None):
280 """Returns the list of configured algs needed to build inputs to jet finding as defined by jetOrConstitdef
282 jetOrConstitdef can either be
283 * a JetDefinition : this happens when called from JetRecCfg or getJetDefAlgs then the jetdef._prereqDic/Order are used.
284 * a JetInputConstit : to allow scheduling the corresponding constituents algs independently of any jet alg.
286 context is only used if jetOrConstitdef is not a JetDefinition and must refer to a context in StandardJetContext.
288 The returned list may contain several algs, including constituent modifications algs, track selection, copying of
289 input truth particles and event density calculations
290 It may also contain ComponentAccumulator, only (?) in reco from RDO/RAW when we need to build externals such as clusters or tracks : in this case we call the main config functions from external packages)
294 from .JetDefinition
import JetInputConstit, JetDefinition
295 if isinstance(jetOrConstitdef, JetInputConstit):
297 jetlog.info(
"Setting up jet inputs from JetInputConstit : "+jetOrConstitdef.name)
298 jetdef =
solveDependencies( JetDefinition(
'Kt', 0., jetOrConstitdef, context=context), flags )
303 jetdef = jetOrConstitdef
305 jetlog.info(
"Inspecting input file contents")
309 filecontents = jetdef._cflags.Input.Collections
314 cname = c.containername
if isinstance(c, JetInputConstit)
else c.containername(jetdef,c.specs)
315 return cname
in filecontents
318 inputdeps = [ inputkey
for inputkey
in jetdef._prereqOrder
if inputkey.startswith(
'input:')
or inputkey.startswith(
'extinput:') ]
320 for inputfull
in inputdeps:
321 inputInstance = jetdef._prereqDic[inputfull]
322 if isInInput( inputInstance ):
323 jetlog.info(f
"Input container for {inputInstance} already in input file.")
327 if isinstance(inputInstance, JetInputConstit):
330 alg = inputInstance.algoBuilder( jetdef, inputInstance.specs )
342 """Construct the name of the PseudoJetContainer defined by the given JetDef or JetInputConstit.
343 This name has to be constructed from various places, so we factorize the definition here.
345 cdef = jetOrConstitdef
if isinstance(jetOrConstitdef, JetInputConstit)
else jetOrConstitdef.inputdef
346 _str_containername = cdef.containername(parent_jetdef).
split(
':')[-1]
if callable(cdef.containername)
else cdef.containername
347 end =
'' if suffix
is None else f
'_{suffix}'
348 return f
'PseudoJet{_str_containername}{end}'
351 """returns a configured PseudoJetAlgorithm which converts the inputs defined by constitdef into fastjet::PseudoJet
353 IMPORTANT : constitdef must have its dependencies solved (i.e. it must result from a solveDependencies() call)
355 the flags argument is TEMPORARY and will be removed once further dev on PseudoJetAlgorithm is done (see comment below)
357 _str_containername = constitdef.containername(parent_jetdef).
split(
':')[-1]
if callable(constitdef.containername)
else constitdef.containername
358 jetlog.debug(
"Getting PseudoJetAlg for label {0} from {1}".
format(constitdef.name,constitdef.inputname))
359 end =
'' if suffix
is None else f
'_{suffix}'
360 full_label = constitdef.label + end
361 pjgalg = CompFactory.PseudoJetAlgorithm(
362 "pjgalg_"+_str_containername+end,
363 InputContainer = _str_containername,
364 OutputContainer =
getPJContName(constitdef, suffix = suffix, parent_jetdef = parent_jetdef),
366 SkipNegativeEnergy=
True,
367 DoByVertex=constitdef.byVertex
374 if flags
is not None:
375 from AthenaConfiguration.Enums
import BeamType
376 pjgalg.UseChargedPV = (flags.Beam.Type == BeamType.Collisions)
379 pjgalg.UseChargedPV=
False
380 pjgalg.UseChargedPUsideband=
True
381 elif suffix ==
'Neut':
382 pjgalg.UseCharged=
False
388 """returns a configured PseudoJetAlgorithm which converts the inputs defined by constitdef into fastjet::PseudoJet
390 The difference for the above is this is dedicated to ghosts which need variations for the Label and the muon segment cases.
392 IMPORTANT : ghostdef must have its dependencies solved (i.e. it must result from a solveDependencies() call)
394 label =
"Ghost"+ghostdef.label
395 _container_name = ghostdef.containername(parentjetdef).
split(
":")[1]
if callable(ghostdef.containername)
else ghostdef.containername
396 _output_cont_name_suffix =
"" if parentjetdef.context ==
"default" or _container_name.endswith(parentjetdef.context)
else (
"_" + parentjetdef.context)
399 InputContainer = _container_name,
400 OutputContainer=
"PseudoJetGhost"+_container_name + _output_cont_name_suffix,
402 SkipNegativeEnergy=
True,
405 pjaclass = CompFactory.PseudoJetAlgorithm
406 if ghostdef.basetype==
"MuonSegment":
408 pjaclass = CompFactory.MuonSegmentPseudoJetAlgorithm
409 kwargs.update( Pt =1e-20 )
410 kwargs.pop(
'SkipNegativeEnergy')
412 pjgalg = pjaclass(
"pjgalg_" + label +
"_" + parentjetdef.context, **kwargs )
416 def getJetRecAlg( jetdef, monTool = None, ftf_suffix = '', extraOutputs = []):
417 """Returns the configured JetRecAlg instance corresponding to jetdef
419 IMPORTANT : jetdef must have its dependencies solved (i.e. it must result from solveDependencies() )
421 pjContNames = jetdef._internalAtt[
'finalPJContainer']
424 "JetAlgorithm": jetdef.algorithm,
425 "JetRadius": jetdef.radius,
426 "PtMin": jetdef.ptmin,
427 "InputPseudoJets": pjContNames,
428 "GhostArea": jetdef.ghostarea,
429 "JetInputType":
int(jetdef.inputdef.jetinputtype),
431 "VariableRMinRadius": jetdef.VRMinRadius,
432 "VariableRMassScale": jetdef.VRMassScale
435 jetname = jetdef.fullname()
437 jclust = CompFactory.JetClustererByVertex(
442 jclust = CompFactory.JetClusterer(
449 jra = CompFactory.JetRecAlg(
450 "jetrecalg_"+jetname+ftf_suffix,
453 OutputContainer = jetname+ftf_suffix,
457 jra.MonTool = monTool
460 jra.ExtraOutputs = extraOutputs + [
461 (
'xAOD::JetContainer',f
'{jetname}.Ghost{ghost}')
for ghost
in jetdef.ghostdefs
467 """Returns a configured JetRecAlg set-up to perform the grooming defined by 'groomdef'
468 ('monTool' is a temporary placeholder, it is expected to be used in the trigger in the future)
470 jetlog.debug(
"Configuring grooming alg \"jetalg_{0}\"".
format(groomdef.fullname()))
474 groomClass = CompFactory.getComp(groomdef.tooltype)
475 groomer = groomClass(groomdef.groomalg,
476 UngroomedJets = groomdef.ungroomeddef.fullname(),
477 ParentPseudoJets = groomdef.ungroomeddef._internalAtt[
'finalPJContainer'],
478 **groomdef.properties)
484 jetname = groomdef.fullname()
485 jra = CompFactory.JetRecAlg(
486 "jetrecalg_"+jetname,
489 OutputContainer = jetname)
492 jra.MonTool = monTool
495 jra.ExtraOutputs = extraOutputs + [
496 (
'xAOD::JetContainer',f
'{jetname}.Ghost{ghost}')
for ghost
in groomdef.ungroomeddef.ghostdefs
503 def getJetCopyAlg(jetsin, jetsoutdef, decorations=[], shallowcopy=True, shallowIO=True, monTool=None):
505 Get a JetRecAlg set up to copy a jet collection and apply mods
506 In this setup we do not resolve dependencies because typically
507 these may be set up already in the original jet collection
508 In future we may wish to add a toggle.
510 The decoration list can be set in order for the decorations
511 (jet moments) on the original jets to be propagated to the
512 copy collection. Beware of circular dependencies!
514 jcopy = CompFactory.JetCopier(
517 DecorDeps=decorations,
518 ShallowCopy=shallowcopy,
523 for mod
in jetsoutdef.modifiers:
525 mods.append(
getModifier(jetsoutdef,moddef,moddef.modspec))
527 jetsoutname = jetsoutdef.fullname()
528 jra = CompFactory.JetRecAlg(
529 "jetrecalg_copy_"+jetsoutname,
532 OutputContainer = jetsoutname)
534 jra.MonTool = monTool
542 """returns a configured JetConstituentModSequence or None if constit.modifiers == []
544 The JetConstituentModSequence is determined by the JetInputConstitSeq constitSeq .
545 However, details of the configuration of the JetConstituentModSequence may depends on which JetDefinition
546 this JetConstituentModSequence is intended for. Thus the function also requires a parentjetdef JetDefinition input
548 IMPORTANT : parentjetdef & constitSeq must have their dependencies solved (i.e. they must result from solveDependencies() )
550 See also getConstitModAlg_nojetdef
555 if not isinstance(constitSeq, JetInputConstitSeq):
return
558 inputtype = constitSeq.basetype
560 sequence = constitSeq.modifiers
565 if constitSeq.inputname == constitSeq.containername:
return
567 for step
in sequence:
568 modInstance = parentjetdef._prereqDic[ f
'cmod:{step}' ]
569 if not modInstance.tooltype:
continue
571 toolclass = getattr( CompFactory, modInstance.tooltype)
574 for k,v
in modInstance.properties.items():
576 modInstance.properties[k ] =
v(parentjetdef, constitSeq )
578 tool = toolclass(modInstance.name,**modInstance.properties)
580 if (inputtype == xAODType.FlowElement
or inputtype == xAODType.ParticleFlow)
and modInstance.tooltype
not in [
"CorrectPFOTool",
"ChargedHadronSubtractionTool"]:
581 tool.IgnoreChargedPFO=
True
582 tool.ApplyToChargedPFO=
False
583 tool.InputType = inputtype
586 sequenceshort =
"".
join(sequence)
587 seqname =
"ConstitMod{0}_{1}".
format(sequenceshort,constitSeq.name)
588 inputcontainer =
str(constitSeq.inputname)
589 outputcontainer =
str(constitSeq.containername)
591 if (inputtype == xAODType.FlowElement
or inputtype == xAODType.ParticleFlow):
594 def chopPFO(thestring):
595 pfostr =
"ParticleFlowObjects"
596 if thestring.endswith(pfostr):
597 return thestring[:-len(pfostr)]
599 inputcontainer = chopPFO(inputcontainer)
600 outputcontainer = chopPFO(outputcontainer)
602 doByVertex = constitSeq.byVertex
604 inChargedFEDecorKeys = []
605 inNeutralFEDecorKeys = []
619 inChargedFEDecorKeys += [
"cellCPData",
"FE_ElectronLinks",
"FE_PhotonLinks",
"FE_MuonLinks"]
620 inNeutralFEDecorKeys += [
"calpfo_NLeadingTruthParticleBarcodeEnergyPairs",
"FE_ElectronLinks",
"FE_PhotonLinks",
"FE_MuonLinks"]
622 modseq = CompFactory.JetConstituentModSequence(seqname,
624 OutputContainer = outputcontainer,
625 InputContainer= inputcontainer,
626 InChargedFEDecorKeys = inChargedFEDecorKeys,
627 InNeutralFEDecorKeys = inNeutralFEDecorKeys,
629 DoByVertex = doByVertex
632 modseq.MonTool = monTool
634 constitmodalg = CompFactory.JetAlgorithm(
"jetalg_{0}".
format(modseq.getName()))
635 constitmodalg.Tools = [modseq]
640 """Same as getConstitModAlg.
641 This is a convenient function to obtain a JetConstituentModSequence when it is certain, no JetDef is needed.
642 This function just builds a dummy JetDefinition then calls getConstitModAlg
643 Needed in the trigger config.
645 jetdef =
solveDependencies( JetDefinition(
'Kt', 0., constitSeq, context=context) , flags)
646 constitSeq = jetdef._prereqDic[
'input:'+constitSeq.name]
651 """returns the list of configured JetModifier tools needed by this jetdef.
652 This is done by instantiating the actual C++ tool as ordered in jetdef._prereqOrder
654 modlist = [ key
for key
in jetdef._prereqOrder
if key.startswith(
'mod:')]
657 for modkey
in modlist:
658 moddef = jetdef._prereqDic[modkey]
660 modspec =
'' if ':' not in modkey
else modkey.split(
':',1)[1]
668 """Translate JetModifier into a concrete tool"""
669 jetlog.verbose(
"Retrieving modifier {0}".
format(
str(moddef)))
671 if flags
is not None:
673 jetdef = jetdef.clone()
674 jetdef._cflags = flags
675 jetdef._contextDic = flags.Jet.Context[jetdef.context]
679 modtool = moddef.createfn(jetdef, modspec)
680 except Exception
as e:
681 jetlog.error( f
"Unhandled modifier specification {modspec} for mod {moddef} acting on jet def {jetdef.basename}!")
682 jetlog.error( f
"Received exception \"{e}\"" )
683 jetlog.error( f
"Helper function is \"{moddef.createfn}\"" )
684 raise ValueError( f
"JetModConfig unable to handle mod {moddef} with spec \"{modspec}\"")
689 for k,v
in moddef.properties.items():
692 v =
v(jetdef, modspec)
693 setattr(modtool, k, v)
701 """Filters the lists jetdef.modifiers and jetdef.ghosts (and jetdef._prereqOrder), so only the components
702 comptatible with flags are selected.
703 If flags==None : assume jetdef._cflags is properly set (this is done by higher-level functions)
704 The compatibility is ultimately tested using the component 'filterfn' attributes.
705 Internally calls the function isComponentPassingConditions() (see below)
707 jetlog.info(
"Standard Reco mode : filtering components in "+
str(jetdef))
709 if jetdef._cflags
is None:
710 jetdef._cflags = flags
721 raise Exception(f
"JetDefinition {jetdef} can NOT be scheduled. Failure of input {jetdef.inputdef.name} reason={reason}" )
722 jetlog.info(f
"IMPORTANT : removing {jetdef} because input incompatible with job conditions. Reason={reason} ")
725 if isinstance( jetdef.inputdef, JetInputConstitSeq):
727 jetdef.inputdef.modifiers =
filterJetDefList(jetdef, jetdef.inputdef.modifiers,
'cmod', raiseOnFailure, jetdef._cflags)
732 jetdef.ghostdefs =
filterJetDefList(jetdef, jetdef.ghostdefs,
"ghost", raiseOnFailure, jetdef._cflags)
733 jetdef.modifiers =
filterJetDefList(jetdef, jetdef.modifiers,
"mod", raiseOnFailure, jetdef._cflags)
742 groomdef.modifiers =
filterJetDefList(groomdef, groomdef.modifiers,
"mod", raiseOnFailure, flags)
752 basekey= compType+
':' if compType!=
"" else ""
754 fullname = jetdef.fullname()
758 fullkey = basekey+comp
759 cInstance = jetdef._prereqDic[fullkey]
763 raise Exception(
"JetDefinition {} can NOT be scheduled. Failure of {} {} reason={}".
format(
764 jetdef, compType, comp, reason) )
767 jetlog.info(f
"{fullname} : removing {compType} {comp} reason={reason}")
768 if fullkey
in jetdef._prereqOrder:
769 jetdef._prereqOrder.remove(fullkey)
770 if compType==
'ghost':
776 jetlog.info(
" *** Number of {} filtered components = {} final list={}".
format(compType, nOut, outList) )
784 """Test if component is compatible with flags.
785 This is done by calling component.filterfn AND testing all its prereqs.
787 for req
in component.prereqs:
788 _str_req = req(jetdef)
if callable(req)
else req
789 if _str_req
not in jetdef._prereqDic:
790 return False,
"prereq "+_str_req+
" not available"
791 reqInstance = jetdef._prereqDic[_str_req]
794 return False,
"prereq "+
str(reqInstance)+
" failed because : "+reason
796 ok, reason = component.filterfn(flags)
801 from AthenaConfiguration.Enums
import Project
802 return Project.determine() in( Project.AnalysisBase, Project.AthAnalysis)
806 """In runIII the scheduler automatically orders algs, so the JetRecConfig helpers do not try to enforce the correct ordering.
807 This is not the case in runII config for which this jobO is intended --> This function makes sure some jet-related algs are well ordered.
809 def _flatten_CA(cfg, sequence_name="AthAlgSeq"):
810 from AthenaConfiguration.ComponentAccumulator
import ConfigurationError
811 if not isinstance(cfg, ComponentAccumulator):
812 raise ConfigurationError(
'It is not allowed to flatten with multiple top sequences')
814 if len(cfg._allSequences) != 1:
815 raise ConfigurationError(
'It is not allowed to flatten with multiple top sequences')
817 sequence = cfg.getSequence(sequence_name)
818 if sequence.Sequential:
819 raise ConfigurationError(
'It is not allowed to flatten sequential sequences')
822 for member
in sequence.Members:
823 if isinstance(member, CompFactory.AthSequencer):
824 members.extend(_flatten_CA(cfg, member.getName()))
826 members.append(member)
828 sequence.Members = members
834 if not isinstance(a, ComponentAccumulator) :
838 ca_algs =
list(a._algorithms.keys())
840 algs_tmp.append(a.popEventAlgo(algo))
844 evtDensityAlgs = [(i, alg)
for (i, alg)
in enumerate(algs)
if alg
and alg.getType() ==
'EventDensityAthAlg' ]
845 pjAlgs = [(i, alg)
for (i, alg)
in enumerate(algs)
if alg
and alg.getType() ==
'PseudoJetAlgorithm' ]
847 for i, edalg
in evtDensityAlgs:
848 edInput = edalg.EventDensityTool.InputContainer
849 for j, pjalg
in pjAlgs:
852 if edInput ==
str(pjalg.OutputContainer):
853 pairsToswap.append((i, j))
854 for i, j
in pairsToswap:
855 algs[i], algs[j] = algs[j], algs[i]
862 """Make the jet collection described by jetdef available as constituents to other jet finding
864 Technically : create JetInputExternal and JetInputConstit and register them in the relevant look-up dictionnaries.
865 the JetInputConstit will have a algoBuilder to generate the JetContainer described by jetdef
867 from .StandardJetConstits
import stdConstitDic, stdInputExtDic
868 jetname = jetdef.fullname()
871 def jetBuilder(largejetdef,spec):
872 return JetRecCfg(largejetdef._cflags, jetdef)
874 stdInputExtDic[jetname] = JetInputExternal( jetname, jetname, algoBuilder=jetBuilder)
875 stdConstitDic[jetname] = JetInputConstit(jetname, xAODType.Jet, jetname )
887 from AthenaConfiguration.AllConfigFlags
import initConfigFlags
890 from AthenaConfiguration.TestDefaults
import defaultTestFiles, defaultConditionsTags
891 flags.Input.Files = defaultTestFiles.LATEST_AOD_MC
892 flags.IOVDb.GlobalTag = defaultConditionsTags.LATEST_MC
897 from AthenaConfiguration.MainServicesConfig
import MainServicesCfg
901 from AthenaPoolCnvSvc.PoolReadConfig
import PoolReadCfg
905 from JetRecConfig.StandardSmallRJets
import AntiKt4EMTopo
906 cfg.merge(
JetRecCfg(flags, AntiKt4EMTopo))
908 cfg.printConfig(withDetails=
False,summariseProps=
True)
911 if __name__==
"__main__":