1#include "Tracking/dqm/TrackingRecoDQM.h"
3namespace tracking::dqm {
6 track_collection_ = parameters.
get<std::string>(
"track_collection");
7 truth_collection_ = parameters.
get<std::string>(
"truth_collection");
8 measurement_collection_ =
9 parameters.
get<std::string>(
"measurement_collection");
10 measurement_passname_ = parameters.
get<std::string>(
"measurement_passname");
12 ecal_sp_coll_name_ = parameters.
get<std::string>(
"ecal_sp_coll_name");
13 ecal_sp_passname_ = parameters.
get<std::string>(
"ecal_sp_passname");
14 target_sp_coll_name_ = parameters.
get<std::string>(
"target_sp_coll_name");
15 target_sp_passname_ = parameters.
get<std::string>(
"target_sp_passname");
16 truth_passname_ = parameters.
get<std::string>(
"truth_passname");
17 truth_events_passname_ = parameters.
get<std::string>(
"truth_events_passname");
18 track_passname_ = parameters.
get<std::string>(
"track_passname");
19 track_collection_events_passname_ =
20 parameters.
get<std::string>(
"track_collection_events_passname");
22 title_ = parameters.
get<std::string>(
"title",
"tagger_trk_");
23 track_prob_cut_ = parameters.
get<
double>(
"trackProb_cut", 0.5);
24 subdetector_ = parameters.
get<std::string>(
"subdetector",
"Tagger");
25 track_states_ = parameters.
get<std::vector<std::string>>(
"track_states", {});
27 pidmap_[-321] = PIDBins::kminus;
28 pidmap_[321] = PIDBins::kplus;
29 pidmap_[-211] = PIDBins::piminus;
30 pidmap_[211] = PIDBins::piplus;
31 pidmap_[11] = PIDBins::electron;
32 pidmap_[-11] = PIDBins::positron;
33 pidmap_[2212] = PIDBins::proton;
34 pidmap_[-2212] = PIDBins::antiproton;
38 ldmx_log(trace) <<
"DQM Reading in:" << track_collection_;
40 if (!event.
exists(track_collection_, track_collection_events_passname_)) {
41 ldmx_log(error) <<
"TrackCollection " << track_collection_
42 <<
" with pass = " << track_collection_events_passname_
48 event.getCollection<
ldmx::Track>(track_collection_, track_passname_)};
50 if (!event.
exists(measurement_collection_, measurement_passname_)) {
51 ldmx_log(error) <<
"Measurement collection " << measurement_collection_
52 <<
" with pass = " << measurement_passname_
58 measurement_collection_, measurement_passname_)};
61 if (event.
exists(truth_collection_, truth_events_passname_)) {
62 truth_track_collection_ = std::make_shared<std::vector<ldmx::Track>>(
63 event.getCollection<
ldmx::Track>(truth_collection_, truth_passname_));
64 do_truth_comparison_ =
true;
68 if (event.
exists(ecal_sp_coll_name_, ecal_sp_passname_)) {
69 ecal_scoring_hits_ = std::make_shared<std::vector<ldmx::SimTrackerHit>>(
74 if (event.
exists(target_sp_coll_name_, target_sp_passname_)) {
75 target_scoring_hits_ = std::make_shared<std::vector<ldmx::SimTrackerHit>>(
77 target_sp_passname_));
80 ldmx_log(debug) <<
"Do truth comparison::" << do_truth_comparison_;
82 if (do_truth_comparison_) {
83 sortTracks(tracks, unique_tracks_, duplicate_tracks_, fake_tracks_);
85 unique_tracks_ = tracks;
88 ldmx_log(debug) <<
"Filling histograms for " << tracks.size() <<
" tracks";
93 if (!unique_tracks_.empty()) {
94 ldmx_log(debug) <<
"Track Monitoring on " << unique_tracks_.size()
96 trackMonitoring(unique_tracks_, measurements, title_,
true,
97 do_truth_comparison_);
101 if (!duplicate_tracks_.empty()) {
102 ldmx_log(debug) <<
"Track Monitoring on " << duplicate_tracks_.size()
104 trackMonitoring(duplicate_tracks_, measurements, title_ +
"dup_",
false,
107 if (!fake_tracks_.empty()) {
108 ldmx_log(debug) <<
"Track Monitoring on " << fake_tracks_.size()
110 trackMonitoring(fake_tracks_, measurements, title_ +
"fake_",
false,
false);
115 ldmx_log(trace) <<
"Track Extrapolation to Ecal Monitoring";
116 if (do_truth_comparison_) {
117 if (std::find(track_states_.begin(), track_states_.end(),
"target") !=
118 track_states_.end()) {
122 if (std::find(track_states_.begin(), track_states_.end(),
"ecal") !=
123 track_states_.end()) {
127 if (std::find(track_states_.begin(), track_states_.end(),
"beamOrigin") !=
128 track_states_.end()) {
134 if (do_truth_comparison_) {
135 ldmx_log(trace) <<
"Technical Efficiency plots";
136 efficiencyPlots(tracks, measurements, title_);
142 ldmx_log(trace) <<
"Clear the vectors";
143 unique_tracks_.clear();
144 duplicate_tracks_.clear();
145 fake_tracks_.clear();
152void TrackingRecoDQM::efficiencyPlots(
153 const std::vector<ldmx::Track>& tracks,
154 const std::vector<ldmx::Measurement>& measurements,
155 const std::string& title) {
158 histograms_.
fill(title +
"truth_N_tracks", truth_track_collection_->size());
159 for (
auto& truth_trk : *(truth_track_collection_)) {
160 auto truth_phi = truth_trk.getPhi();
161 auto truth_d0 = truth_trk.getD0();
162 auto truth_z0 = truth_trk.getZ0();
163 auto truth_theta = truth_trk.getTheta();
164 auto truth_qop = truth_trk.getQoP();
165 auto truth_p = 1000. / abs(truth_trk.getQoP());
166 auto truth_n_hits = truth_trk.getNhits();
168 auto truth_mom = truth_trk.getMomentumAtTarget();
169 double truth_pt_beam{0.}, truth_beam_angle{0.};
170 if (truth_mom.size() == 3) {
172 std::sqrt(truth_mom[0] * truth_mom[0] + truth_mom[1] * truth_mom[1]);
173 truth_beam_angle = std::atan2(truth_pt_beam, truth_mom[2]);
185 if (pidmap_.count(truth_trk.getPdgID()) != 0) {
190 if (pidmap_[truth_trk.getPdgID()] == PIDBins::kminus) {
194 if (pidmap_[truth_trk.getPdgID()] == PIDBins::kplus) {
198 if (pidmap_[truth_trk.getPdgID()] == PIDBins::piminus) {
202 if (pidmap_[truth_trk.getPdgID()] == PIDBins::piplus) {
206 if (pidmap_[truth_trk.getPdgID()] == PIDBins::electron) {
210 if (pidmap_[truth_trk.getPdgID()] == PIDBins::positron) {
214 if (pidmap_[truth_trk.getPdgID()] == PIDBins::proton) {
221 for (
auto& track : tracks) {
225 auto it = std::find_if(truth_track_collection_->begin(),
226 truth_track_collection_->end(),
228 return tt.getTrackID() == track.getTrackID();
231 double track_truth_prob = track.getTruthProb();
233 if (it != truth_track_collection_->end() &&
234 track_truth_prob >= track_prob_cut_)
238 if (!truth_trk)
return;
240 auto truth_phi = truth_trk->getPhi();
241 auto truth_d0 = truth_trk->getD0();
242 auto truth_z0 = truth_trk->getZ0();
243 auto truth_theta = truth_trk->getTheta();
244 auto truth_qop = truth_trk->getQoP();
245 auto truth_p = 1000. / abs(truth_trk->getQoP());
248 double truth_pt_beam{0.}, truth_beam_angle{0.};
249 if (truth_mom.size() == 3) {
251 std::sqrt(truth_mom[0] * truth_mom[0] + truth_mom[1] * truth_mom[1]);
252 truth_beam_angle = std::atan2(truth_pt_beam, truth_mom[2]);
265 auto dedx_measurements = track.getDedxMeasurements();
266 auto measurement_idxs = track.getMeasurementsIdxs();
267 for (
size_t i = 0; i < measurement_idxs.size(); ++i) {
269 measurements.at(measurement_idxs[i]).getLayer());
271 if (i < dedx_measurements.size()) {
273 dedx_measurements[i]);
279 if (pidmap_.count(truth_trk->getPdgID()) != 0) {
284 if (pidmap_[truth_trk->getPdgID()] == PIDBins::kminus) {
288 if (pidmap_[truth_trk->getPdgID()] == PIDBins::kplus) {
292 if (pidmap_[truth_trk->getPdgID()] == PIDBins::piminus) {
296 if (pidmap_[truth_trk->getPdgID()] == PIDBins::piplus) {
300 if (pidmap_[truth_trk->getPdgID()] == PIDBins::electron) {
304 if (pidmap_[truth_trk->getPdgID()] == PIDBins::positron) {
308 if (pidmap_[truth_trk->getPdgID()] == PIDBins::proton) {
316void TrackingRecoDQM::trackMonitoring(
317 const std::vector<ldmx::Track>& tracks,
318 const std::vector<ldmx::Measurement>& measurements,
const std::string title,
319 const bool& doDetail,
const bool& doTruth) {
320 for (
auto& track : tracks) {
322 auto trk_d0 = track.getD0();
323 auto trk_z0 = track.getZ0();
324 auto trk_qop = track.getQoP();
325 auto trk_theta = track.getTheta();
326 auto trk_phi = track.getPhi();
327 auto trk_p = 1000. / abs(trk_qop);
328 auto dedx_measurements = track.getDedxMeasurements();
329 auto measurement_idxs = track.getMeasurementsIdxs();
330 for (
size_t i = 0; i < measurement_idxs.size(); ++i) {
332 measurements.at(measurement_idxs[i]).getLayer());
334 if (i < dedx_measurements.size()) {
348 if (title == title_) {
349 const auto& sm_loc0 = track.getSmoothedLoc0();
350 const auto& sm_cov = track.getSmoothedCovLoc0();
351 for (
size_t i = 0; i < measurement_idxs.size(); ++i) {
352 if (i >= sm_loc0.size())
break;
353 const auto& meas = measurements.at(measurement_idxs[i]);
354 int layer = meas.getLayer();
355 float meas_u = meas.getLocalPosition()[0];
356 float V = meas.getLocalCovariance()[0];
359 if (denom <= 0.f)
continue;
360 float r_smooth = meas_u - sm_loc0[i];
361 float res_ubs = r_smooth * V / denom;
362 float pull_ubs = r_smooth / std::sqrt(denom);
370 auto trk_mom = track.getMomentumAtTarget();
372 double px_ldmx{0.}, py_ldmx{0.}, pz_ldmx{0.};
373 double pt_bending{0.}, pt_beam{0.};
374 if (trk_mom.size() == 3) {
375 px_ldmx = trk_mom[0];
376 py_ldmx = trk_mom[1];
377 pz_ldmx = trk_mom[2];
379 pt_bending = std::sqrt(px_ldmx * px_ldmx + pz_ldmx * pz_ldmx);
381 pt_beam = std::sqrt(px_ldmx * px_ldmx + py_ldmx * py_ldmx);
385 Acts::BoundSquareMatrix cov =
386 tracking::sim::utils::unpackCov(track.getPerigeeCov());
388 double sigmad0 = sqrt(
389 cov(Acts::BoundIndices::eBoundLoc0, Acts::BoundIndices::eBoundLoc0));
390 double sigmaz0 = sqrt(
391 cov(Acts::BoundIndices::eBoundLoc1, Acts::BoundIndices::eBoundLoc1));
393 sqrt(cov(Acts::BoundIndices::eBoundPhi, Acts::BoundIndices::eBoundPhi));
394 double sigmatheta = sqrt(
395 cov(Acts::BoundIndices::eBoundTheta, Acts::BoundIndices::eBoundTheta));
396 double sigmaqop = sqrt(cov(Acts::BoundIndices::eBoundQOverP,
397 Acts::BoundIndices::eBoundQOverP));
399 (1000. / trk_qop) * (1000. / trk_qop) * sigmaqop / 1000.;
420 track.getChi2() / track.getNdf());
435 if (track.getNhits() == 8)
437 else if (track.getNhits() == 9)
439 else if (track.getNhits() == 10)
447 auto it = std::find_if(truth_track_collection_->begin(),
448 truth_track_collection_->end(),
450 return tt.getTrackID() == track.getTrackID();
453 double track_truth_prob = track.getTruthProb();
455 if (it != truth_track_collection_->end() &&
456 track_truth_prob >= track_prob_cut_)
461 auto truth_d0 = truth_trk->getD0();
462 auto truth_z0 = truth_trk->getZ0();
463 auto truth_phi = truth_trk->getPhi();
464 auto truth_theta = truth_trk->getTheta();
465 auto truth_qop = truth_trk->getQoP();
466 auto truth_p = 1000. / abs(truth_trk->getQoP());
469 double truth_pt_beam{0.};
470 if (truth_mom.size() == 3) {
471 truth_pt_beam = std::sqrt(truth_mom[0] * truth_mom[0] +
472 truth_mom[1] * truth_mom[1]);
482 double res_d0 = trk_d0 - truth_d0;
483 double res_z0 = trk_z0 - truth_z0;
484 double res_phi = trk_phi - truth_phi;
485 double res_theta = trk_theta - truth_theta;
486 double res_qop = trk_qop - truth_qop;
487 double res_p = trk_p - truth_p;
488 double res_pt_beam = pt_beam - truth_pt_beam;
498 double pull_d0 = res_d0 / sigmad0;
499 double pull_z0 = res_z0 / sigmaz0;
500 double pull_phi = res_phi / sigmaphi;
501 double pull_theta = res_theta / sigmatheta;
502 double pull_qop = res_qop / sigmaqop;
503 double pull_p = res_p / sigmap;
528 if (track.getNhits() == 8)
530 else if (track.getNhits() == 9)
532 else if (track.getNhits() == 10)
545 const std::vector<ldmx::Track>& tracks, ldmx::TrackStateType ts_type,
546 const std::string& ts_title) {
547 for (
auto& track : tracks) {
551 auto it = std::find_if(truth_track_collection_->begin(),
552 truth_track_collection_->end(),
554 return tt.getTrackID() == track.getTrackID();
557 double track_truth_prob = track.getTruthProb();
559 if (it != truth_track_collection_->end() &&
560 track_truth_prob >= track_prob_cut_)
564 if (!truth_trk)
continue;
566 auto trk_ts = track.getTrackState(ts_type);
567 auto truth_ts = truth_trk->getTrackState(ts_type);
569 if (!trk_ts.has_value())
continue;
570 if (!truth_ts.has_value())
continue;
576 if (target_state.pos_mom_cov_.size() < 21)
continue;
580 double sigmaloc0 = std::sqrt(target_state.pos_mom_cov_[0]);
581 double sigmaloc1 = std::sqrt(target_state.pos_mom_cov_[6]);
583 double trk_qop = track.getQoP();
584 double trk_p = 1000. / abs(trk_qop);
587 double track_state_loc0 = target_state.pos_[0];
588 double track_state_loc1 = target_state.pos_[1];
590 double truth_state_loc0 = truth_target_state.pos_[0];
591 double truth_state_loc1 = truth_target_state.pos_[1];
593 histograms_.
fill(title_ +
"trk_" + ts_title +
"_loc0", track_state_loc0);
594 histograms_.
fill(title_ +
"trk_" + ts_title +
"_loc1", track_state_loc1);
600 title_ +
"trk_" + ts_title +
"_loc0-truth_" + ts_title +
"_loc0",
601 track_state_loc0 - truth_state_loc0);
603 title_ +
"trk_" + ts_title +
"_loc1-truth_" + ts_title +
"_loc1",
604 track_state_loc1 - truth_state_loc1);
608 (track_state_loc0 - truth_state_loc0) / sigmaloc0);
610 (track_state_loc1 - truth_state_loc1) / sigmaloc1);
616 track.getNhits(), track_state_loc0 - truth_state_loc0);
618 track.getNhits(), track_state_loc1 - truth_state_loc1);
623 (track_state_loc0 - truth_state_loc0) / sigmaloc0);
626 (track_state_loc1 - truth_state_loc1) / sigmaloc1);
630 track_state_loc0 - truth_state_loc0);
632 track_state_loc1 - truth_state_loc1);
636 (track_state_loc0 - truth_state_loc0) / sigmaloc0);
638 (track_state_loc1 - truth_state_loc1) / sigmaloc1);
643void TrackingRecoDQM::sortTracks(
const std::vector<ldmx::Track>& tracks,
644 std::vector<ldmx::Track>& uniqueTracks,
645 std::vector<ldmx::Track>& duplicateTracks,
646 std::vector<ldmx::Track>& fakeTracks) {
648 std::vector<ldmx::Track> sorted_tracks = tracks;
651 std::sort(sorted_tracks.begin(), sorted_tracks.end(),
653 return t1.getTrackID() < t2.getTrackID();
657 for (
size_t i = 0; i < sorted_tracks.size(); i++) {
658 if (sorted_tracks[i].getTruthProb() < track_prob_cut_)
659 fakeTracks.push_back(sorted_tracks[i]);
663 if (uniqueTracks.size() == 0 ||
664 sorted_tracks[i].getTrackID() != sorted_tracks[i - 1].getTrackID()) {
665 uniqueTracks.push_back(sorted_tracks[i]);
671 else if (sorted_tracks[i].getTruthProb() >
672 uniqueTracks.back().getTruthProb()) {
673 duplicateTracks.push_back(uniqueTracks.back());
674 uniqueTracks.back() = sorted_tracks[i];
678 duplicateTracks.push_back(sorted_tracks[i]);
685 if (uniqueTracks.size() + duplicateTracks.size() + fakeTracks.size() !=
687 std::cerr <<
"Error: unique and duplicate tracks vectors do not add up to "
688 "original tracks vector";
693 ldmx_log(trace) <<
"Unique tracks:";
695 ldmx_log(trace) <<
"\tTrack ID: " << track.getTrackID()
696 <<
", Truth Prob: " << track.getTruthProb();
698 ldmx_log(trace) <<
"Duplicate tracks:";
700 ldmx_log(trace) <<
"\tTrack ID: " << track.getTrackID()
701 <<
", Truth Prob: " << track.getTruthProb();
703 ldmx_log(trace) <<
"Fake tracks:";
705 ldmx_log(trace) <<
"\tTrack ID: " << track.getTrackID()
706 <<
", Truth Prob: " << track.getTruthProb();
#define DECLARE_ANALYZER(CLASS)
Macro which allows the framework to construct an analyzer given its name during configuration.
HistogramPool histograms_
helper object for making and filling histograms
Implements an event buffer system for storing event data.
bool exists(const std::string &name, const std::string &passName, bool unique=true) const
Check for the existence of an object or collection with the given name and pass name in the event.
void fill(const std::string &name, const T &val)
Fill a 1D histogram.
Class encapsulating parameters for configuring a processor.
const T & get(const std::string &name) const
Retrieve the parameter of the given name.
Represents a simulated tracker hit in the simulation.
Implementation of a track object.
std::vector< double > getMomentumAtTarget() const
Returns the momentum (px, py, pz) in MeV in the LDMX global frame from the AtTarget TrackState.
void trackStateMonitoring(const std::vector< ldmx::Track > &tracks, ldmx::TrackStateType ts_type, const std::string &ts_title)
Monitoring plots for tracks extrapolated to the ECAL Scoring plane.
void onProcessEnd() override
Callback for the EventProcessor to take any necessary action when the processing of events finishes,...
void configure(framework::config::Parameters ¶meters) override
Configure the analyzer using the given user specified parameters.
void analyze(const framework::Event &event) override
Process the event and make histograms or summaries.