CMS 3D CMS Logo

PFTauPrimaryVertexProducer.cc
Go to the documentation of this file.
1 /* class PFTauPrimaryVertexProducer
2  * EDProducer of the
3  * authors: Ian M. Nugent
4  * This work is based on the impact parameter work by Rosamaria Venditti and reconstructing the 3 prong taus.
5  * The idea of the fully reconstructing the tau using a kinematic fit comes from
6  * Lars Perchalla and Philip Sauerland Theses under Achim Stahl supervision. This
7  * work was continued by Ian M. Nugent and Vladimir Cherepanov.
8  * Thanks goes to Christian Veelken and Evan Klose Friis for their help and suggestions.
9  */
10 
11 
21 
27 
42 
46 
49 #include <memory>
50 #include <TFormula.h>
51 
52 #include <memory>
53 
54 using namespace reco;
55 using namespace edm;
56 using namespace std;
57 
59  public:
60  enum Alg{useInputPV=0, useFontPV};
61 
62  struct DiscCutPair{
63  DiscCutPair():discr_(nullptr),cutFormula_(nullptr){}
64  ~DiscCutPair(){delete cutFormula_;}
67  double cut_;
68  TFormula* cutFormula_;
69  };
70  typedef std::vector<DiscCutPair*> DiscCutPairVec;
71 
72  explicit PFTauPrimaryVertexProducer(const edm::ParameterSet& iConfig);
73  ~PFTauPrimaryVertexProducer() override;
74  void produce(edm::Event&,const edm::EventSetup&) override;
75 
76  private:
93  DiscCutPairVec discriminators_;
94  std::unique_ptr<StringCutObjectSelector<reco::PFTau> > cut_;
95  std::unique_ptr<tau::RecoTauVertexAssociator> vertexAssociator_;
96 };
97 
99  PFTauTag_(iConfig.getParameter<edm::InputTag>("PFTauTag")),
100  PFTauToken_(consumes<std::vector<reco::PFTau> >(PFTauTag_)),
101  ElectronTag_(iConfig.getParameter<edm::InputTag>("ElectronTag")),
102  ElectronToken_(consumes<std::vector<reco::Electron> >(ElectronTag_)),
103  MuonTag_(iConfig.getParameter<edm::InputTag>("MuonTag")),
104  MuonToken_(consumes<std::vector<reco::Muon> >(MuonTag_)),
105  PVTag_(iConfig.getParameter<edm::InputTag>("PVTag")),
106  PVToken_(consumes<reco::VertexCollection>(PVTag_)),
107  beamSpotTag_(iConfig.getParameter<edm::InputTag>("beamSpot")),
108  beamSpotToken_(consumes<reco::BeamSpot>(beamSpotTag_)),
109  Algorithm_(iConfig.getParameter<int>("Algorithm")),
110  qualityCutsPSet_(iConfig.getParameter<edm::ParameterSet>("qualityCuts")),
111  useBeamSpot_(iConfig.getParameter<bool>("useBeamSpot")),
112  useSelectedTaus_(iConfig.getParameter<bool>("useSelectedTaus")),
113  RemoveMuonTracks_(iConfig.getParameter<bool>("RemoveMuonTracks")),
114  RemoveElectronTracks_(iConfig.getParameter<bool>("RemoveElectronTracks"))
115 {
117  std::vector<edm::ParameterSet> discriminators =iConfig.getParameter<std::vector<edm::ParameterSet> >("discriminators");
118  // Build each of our cuts
119  for(auto const& pset : discriminators) {
120  DiscCutPair* newCut = new DiscCutPair();
121  newCut->inputToken_ =consumes<reco::PFTauDiscriminator>(pset.getParameter<edm::InputTag>("discriminator"));
122 
123  if ( pset.existsAs<std::string>("selectionCut") ) newCut->cutFormula_ = new TFormula("selectionCut", pset.getParameter<std::string>("selectionCut").data());
124  else newCut->cut_ = pset.getParameter<double>("selectionCut");
125  discriminators_.push_back(newCut);
126  }
127  // Build a string cut if desired
128  if (iConfig.exists("cut")) cut_.reset(new StringCutObjectSelector<reco::PFTau>(iConfig.getParameter<std::string>( "cut" )));
130  produces<edm::AssociationVector<PFTauRefProd, std::vector<reco::VertexRef> > >();
131  produces<VertexCollection>("PFTauPrimaryVertices");
132 
133  vertexAssociator_.reset(new tau::RecoTauVertexAssociator(qualityCutsPSet_,consumesCollector()));
134 }
135 
137 
139  // Obtain Collections
140  edm::ESHandle<TransientTrackBuilder> transTrackBuilder;
141  iSetup.get<TransientTrackRecord>().get("TransientTrackBuilder",transTrackBuilder);
142 
144  iEvent.getByToken(PFTauToken_,Tau);
145 
147  iEvent.getByToken(ElectronToken_,Electron);
148 
150  iEvent.getByToken(MuonToken_,Mu);
151 
153  iEvent.getByToken(PVToken_,PV);
154 
156  iEvent.getByToken(beamSpotToken_,beamSpot);
157 
158  // Set Association Map
159  auto AVPFTauPV = std::make_unique<edm::AssociationVector<PFTauRefProd, std::vector<reco::VertexRef>>>(PFTauRefProd(Tau));
160  auto VertexCollection_out = std::make_unique<VertexCollection>();
161  reco::VertexRefProd VertexRefProd_out = iEvent.getRefBeforePut<reco::VertexCollection>("PFTauPrimaryVertices");
162 
163  // Load each discriminator
164  for(auto& disc : discriminators_) {
166  iEvent.getByToken(disc->inputToken_, discr);
167  disc->discr_ = &(*discr);
168  }
169 
170  // Set event for VerexAssociator if needed
172  vertexAssociator_->setEvent(iEvent);
173 
174  // For each Tau Run Algorithim
175  if(Tau.isValid()){
176  for(reco::PFTauCollection::size_type iPFTau = 0; iPFTau < Tau->size(); iPFTau++) {
177  reco::PFTauRef tau(Tau, iPFTau);
178  reco::Vertex thePV;
179  if(useInputPV==Algorithm_){
180  thePV =(*( vertexAssociator_->associatedVertex(*tau)));
181  }
182  else if(useFontPV==Algorithm_){
183  thePV=PV->front();
184  }
186  // Check if it passed all the discrimiantors
187  bool passed(true);
188  for(auto const& disc : discriminators_) {
189  // Check this discriminator passes
190  bool passedDisc = true;
191  if ( disc->cutFormula_ )passedDisc = (disc->cutFormula_->Eval((*disc->discr_)[tau]) > 0.5);
192  else passedDisc = ((*disc->discr_)[tau] > disc->cut_);
193  if ( !passedDisc ){passed = false; break;}
194  }
195  if (passed && cut_.get()){passed = (*cut_)(*tau);}
196  if (passed){
197  std::vector<reco::TrackBaseRef> SignalTracks;
198  for(reco::PFTauCollection::size_type jPFTau = 0; jPFTau < Tau->size(); jPFTau++) {
199  if(useSelectedTaus_ || iPFTau==jPFTau){
200  reco::PFTauRef RefPFTau(Tau, jPFTau);
202  // Get tracks from PFTau daugthers
203  const std::vector<edm::Ptr<reco::PFCandidate> > cands = RefPFTau->signalPFChargedHadrCands();
204  for (std::vector<edm::Ptr<reco::PFCandidate> >::const_iterator iter = cands.begin(); iter!=cands.end(); iter++){
205  if(iter->get()->trackRef().isNonnull()) SignalTracks.push_back(reco::TrackBaseRef(iter->get()->trackRef()));
206  else if(iter->get()->gsfTrackRef().isNonnull()){SignalTracks.push_back(reco::TrackBaseRef(((iter)->get()->gsfTrackRef())));}
207  }
208  }
209  }
210  // Get Muon tracks
211  if(RemoveMuonTracks_){
212 
213  if(Mu.isValid()) {
214  for(reco::MuonCollection::size_type iMuon = 0; iMuon< Mu->size(); iMuon++){
215  reco::MuonRef RefMuon(Mu, iMuon);
216  if(RefMuon->track().isNonnull()) SignalTracks.push_back(reco::TrackBaseRef(RefMuon->track()));
217  }
218  }
219  }
220  // Get Electron Tracks
222  if(Electron.isValid()) {
223  for(reco::ElectronCollection::size_type iElectron = 0; iElectron<Electron->size(); iElectron++){
224  reco::ElectronRef RefElectron(Electron, iElectron);
225  if(RefElectron->track().isNonnull()) SignalTracks.push_back(reco::TrackBaseRef(RefElectron->track()));
226  }
227  }
228  }
230  // Get Non-Tau tracks
231  reco::TrackCollection nonTauTracks;
232  for(std::vector<reco::TrackBaseRef>::const_iterator vtxTrkRef=thePV.tracks_begin();vtxTrkRef<thePV.tracks_end();vtxTrkRef++){
233  bool matched = false;
234  for (unsigned int sigTrk = 0; sigTrk < SignalTracks.size(); sigTrk++) {
235  if ( (*vtxTrkRef) == SignalTracks[sigTrk] ) {
236  matched = true;
237  }
238  }
239  if ( !matched ) nonTauTracks.push_back(**vtxTrkRef);
240  }
242  // Refit the vertex
243  TransientVertex transVtx;
244  std::vector<reco::TransientTrack> transTracks;
245  for (reco::TrackCollection::iterator iter=nonTauTracks.begin(); iter!=nonTauTracks.end(); ++iter){
246  transTracks.push_back(transTrackBuilder->build(*iter));
247  }
248  bool FitOk(true);
249  if ( transTracks.size() >= 2 ) {
251  avf.setWeightThreshold(0.1); //weight per track. allow almost every fit, else --> exception
252  try {
253  if ( !useBeamSpot_ ){
254  transVtx = avf.vertex(transTracks);
255  } else {
256  transVtx = avf.vertex(transTracks, *beamSpot);
257  }
258  } catch (...) {
259  FitOk = false;
260  }
261  } else FitOk = false;
262  if ( FitOk ) thePV = transVtx;
263  }
264  VertexRef VRef = reco::VertexRef(VertexRefProd_out, VertexCollection_out->size());
265  VertexCollection_out->push_back(thePV);
266  AVPFTauPV->setValue(iPFTau, VRef);
267  }
268  }
269  iEvent.put(std::move(VertexCollection_out),"PFTauPrimaryVertices");
270  iEvent.put(std::move(AVPFTauPV));
271 }
272 
void produce(edm::Event &, const edm::EventSetup &) override
T getParameter(std::string const &) const
OrphanHandle< PROD > put(std::unique_ptr< PROD > product)
Put a new product.
Definition: Event.h:137
bool isNonnull() const
Checks for non-null.
Definition: Ref.h:253
trackRef_iterator tracks_end() const
last iterator over tracks
Definition: Vertex.cc:81
edm::EDGetTokenT< reco::BeamSpot > beamSpotToken_
bool getByToken(EDGetToken token, Handle< PROD > &result) const
Definition: Event.h:579
#define DEFINE_FWK_MODULE(type)
Definition: MakerMacros.h:17
edm::EDGetTokenT< reco::PFTauDiscriminator > inputToken_
edm::EDGetTokenT< std::vector< reco::PFTau > > PFTauToken_
std::vector< Track > TrackCollection
collection of Tracks
Definition: TrackFwd.h:14
reco::TransientTrack build(const reco::Track *p) const
bool exists(std::string const &parameterName) const
checks if a parameter exists
CachingVertex< 5 > vertex(const std::vector< reco::TransientTrack > &) const override
std::vector< Vertex > VertexCollection
collection of Vertex objects
Definition: VertexFwd.h:9
#define nullptr
edm::RefProd< PFTauCollection > PFTauRefProd
references to PFTau collection
Definition: PFTauFwd.h:15
uint16_t size_type
edm::EDGetTokenT< reco::VertexCollection > PVToken_
int iEvent
Definition: GenABIO.cc:230
Definition: Muon.py:1
edm::EDGetTokenT< std::vector< reco::Muon > > MuonToken_
bool isValid() const
Definition: HandleBase.h:74
edm::Ref< VertexCollection > VertexRef
persistent reference to a Vertex
Definition: VertexFwd.h:13
RefProd< PROD > getRefBeforePut()
Definition: Event.h:167
Definition: L1GtObject.h:30
std::vector< DiscCutPair * > DiscCutPairVec
std::unique_ptr< StringCutObjectSelector< reco::PFTau > > cut_
fixed size matrix
HLT enums.
PFTauPrimaryVertexProducer(const edm::ParameterSet &iConfig)
T get() const
Definition: EventSetup.h:68
std::unique_ptr< tau::RecoTauVertexAssociator > vertexAssociator_
trackRef_iterator tracks_begin() const
first iterator over tracks
Definition: Vertex.cc:76
def move(src, dest)
Definition: eostools.py:511
edm::EDGetTokenT< std::vector< reco::Electron > > ElectronToken_