One question everyone will have is: how do I know what physics object
information/variables are actually stored in my xAOD for each type? You
can be sure for “particles” (inheriting from
IParticle
)
you will have things like pT, η, and φ. But what other
variables are associated to the different object types? We’ll try to answer
that question…
A “container” or “collection” is just a group of objects of the same
type - similar to a vector
in c++ (with some extra fancy tricks).
In order to “retrieve” the information stored in the xAOD containers
we need to know the container type and the container key name. We will
use a handy script called checkxAOD.py
. If you have an xAOD file
called xAOD.pool.root
and want to know the containers and associated
key names, do the following:
checkxAOD.py xAOD.pool.root
Note: You need to replace the fake xAOD.pool.root
with the full
path to an xAOD sample, for example:
checkxAOD.py $ALRB_TutorialData/mc20_13TeV.312276.aMcAtNloPy8EG_A14N30NLO_LQd_mu_ld_0p3_beta_0p5_2ndG_M1000.deriv.DAOD_PHYS.e7587_a907_r14861_p6117/DAOD_PHYS.37791038._000001.pool.root.1
Alternatively you can make a symbolic link using:
ln -s $ALRB_TutorialData/mc20_13TeV.312276.aMcAtNloPy8EG_A14N30NLO_LQd_mu_ld_0p3_beta_0p5_2ndG_M1000.deriv.DAOD_PHYS.e7587_a907_r14861_p6117/DAOD_PHYS.37791038._000001.pool.root.1 xAOD.pool.root
checkxAOD.py xAOD.pool.root
If you have not set
ALRB_TutorialData
in your current shell nor set it in your.bashrc
file, you will need to set it before using it. For more details, see xAOD Samples.
The last column will show you the xAOD container names and types. When
you are retrieving information you usually need to know the container
type (for example xAOD::ElectronContainer
) and the key name for the
particular instance of that container you are interested in (for example
"Electrons"
). In your analysis you can ignore the Aux
containers
(for Auxiliary store), these hold some behind-the-scenes magic. You can
also “mostly” ignore the versions like _v1
. Most information in the
xAOD is stored and retrieved via the Auxiliary store. The user doesn’t
need to worry about this Auxiliary store, and only retrieves the interface
from the event store (e.g. TEvent
for ROOT standalone analysis). So
now you should know the container type and key name. If you use the
wrong key name the code will compile, but it will fail at run-time.
If you run the
checkxAOD.py
script on the file above (and generally in any file) you will see that the type printed for the"Electrons"
container is actuallyDataVector<xAOD::Electron_v1>
. ThexAOD::ElectronContainer
name is an alias we use for theDataVector<xAOD::Electron_v1>
name in the file. Those aliases are centrally defined and one should always use the alias. If you struggle with the translation, here is some rough guidance: In the file every object has a version suffix (_v1
) that we don’t use, that should be ignored, and that is not part of the alias. In the file theContainer
objects are stored asDataVector<...>
, but for the alias we replace theDataVector
part by simply addingContainer
to the object name.
The xAOD containers store xAOD objects. Each xAOD object has a set of associated variables that are part of its class definition. You can find documentation of all xAOD classes here.
In addition to the class variables, additional variables can be added as “decorations” to object containers (and can be accessed from individual objects). You can think of these as extra variables that are attached to the objects but might not always be there. These variables can be seen directly using ROOT. To do this, make sure you have an analysis release set up and do:
root -l xAOD.pool.root
root [1] CollectionTree->Print("Electrons*")
You will get a printout of all variables you can access from that container (aka collection). The variable name you will use in your code is the one that comes after the “.”. For example, you might see:
ElectronsAuxDyn.charge : vector<float>
In your analysis code (after setting up the interface magic that
will be discussed shortly), you can access this variable from the
xAOD::Electron
object by calling charge()
.
Because decorations may or may not be there, it’s good practice to check if they are available before trying to access them, and to print some messages if you need them and they are unavailable.