layout | title | subtitle | minutes |
---|---|---|---|
page |
First steps in LHCb |
Replace a mass hypothesis |
10 |
- Create a new selection starting from the stripping line output
- Change the particle hypothesis made by the stripping line
There are many situations where you want to repurpose a stripping line to look for a new decay that is similar in topology but distinct from what was put into the stripping line. The easiest thing to do is to change some of the hypothesis on particle IDs made in the stripping. This lesson will show you how to do that.
Note that with this method you can never find more candidates than the stripping has already found. However, you can reinterpret parts of the decay to look for new decay modes.
As an example we will switch the decay of the D0 from (K pi) to (pi pi).
There is an algorithm that allows us to replace parts of the decay descriptor called SubstitutePID
:
# configure an algorithm to substitute the Kaon in the D0-decay by a second pion
from Configurables import SubstitutePID
subs = SubstitutePID(
'MakeD02pipi',
Code = "DECTREE('[D*(2010)+ -> (D0 -> K- pi+) pi+]CC')",
# note that SubstitutePID can't handle automatic CC
Substitutions = {
'Charm -> (D0 -> ^K- pi+) Meson': 'pi-',
'Charm -> (D0 -> ^K+ pi-) Meson': 'pi+'
}
)
The algorithm is configured with a name MakeD02pipi
. In the Code
argument we need to specify the initial selection. This is done by using LoKi functors. Since we know we will be using an already prepared selection, we can simply use the DECTREE
functor to search for candidates fulfilling this decay structure. See the lesson on LoKi for more info on what you can do here.
Now we ware ready to specify which hypotheses to change. Substitutions
is a dictionary where the keys are decay descriptors and the values are the names of the replacement particles. The particle that should be replaced is marked with a ^
. So in the example above
'Charm -> (D0 -> ^K- pi+) Meson': 'pi-'
means: Look for a decay of a Charm-particle into D0 plus any meson, where the D0 decays to (K- pi+) and replace the K- with a pi-.
Note that SubstitutePID
does not automatically handle complex conjugation via the CC
operator. Therefore you have to specify all substitutions explicitely.
Next we have to handle the input and output of this algorithm. This is accomplished using particle selections. The input to our substitution algorithm will be the candidates produced by the stripping. In order to make them look like a selection we can use the DataOnDemand
service:
from PhysSelPython.Wrappers import Selection
from PhysSelPython.Wrappers import SelectionSequence
from PhysSelPython.Wrappers import DataOnDemand
# Stream and stripping line we want to use
stream = 'AllStreams'
line = 'D2hhCompleteEventPromptDst2D2RSLine'
tesLoc = '/Event/{0}/Phys/{1}/Particles'.format(stream, line)
# get the selection(s) created by the stripping
strippingSels = [DataOnDemand(Location=tesLoc)]
The output of the algorithm has to be packaged into a new selection:
# create a selection using the substitution algorithm
selSub = Selection(
'Dst2D0pi_D02pipi_Sel',
Algorithm=Subs,
RequiredSelections=strippingSels
)
Note how the input stripping selection is daisy-chained to the output selection through the RequiredSelections
(it has to be a list of selections) argument. The new selection is added into a SelectionSequnce
for further use by DaVinci:
selSeq = SelectionSequence('SelSeq', TopSelection=selSub)
We are now ready to produce an ntuple on our newly created selection. As usual we configure a DecayTreeTuple
, which now is looking for the candidates, which have the redefined D0 decay:
# Create an ntuple to capture D*+ decays from the new selection
dtt = DecayTreeTuple('TupleDstToD0pi_D0Topipi')
dtt.Inputs = [selSeq.outputLocation()]
# note the redefined decay of the D0
dtt.Decay = '[D*(2010)+ -> ^(D0 -> ^pi- ^pi+) ^pi+]CC'
The input to the DecayTreeTuple
is taken as the outputLocations
of the SelectionSequence
we just created.
Selections and SelectionSequences are an elegant way to organize the required algorithms that perform the job of selecting data. The particle selection toolkit uses python tricks to make the management of even complicated sequences of selections rather straight forward. In particular the toolkit ensures that all necessary algorithms are run in the correct order to produce the desired selections. It is therfore mainly a tool to manage dependencies. For technical details have a look at the TWiki page.
Finally we add the SelectionSequence
and the DecayTreeTuple
to the DaVinci application. Since we are adding more than one algorithm we need a GaudiSequencer
that takes care of calling everything in the right order:
# add our new selection and the tuple into the sequencer
seq = GaudiSequencer('MyTupleSeq')
seq.Members += [selSeq.sequence()]
seq.Members += [dtt]
DaVinci().appendToMainSequence([seq])
The solution to this exercise ntuple_switchHypo.py
, is available
here.
Why can't we use this method to look for D* decaying to D0+photon ?