forked from acts-project/acts
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
refactor: Central truth matching for tracks in Examples (acts-project…
…#2904) An attempt to move to a central truth matching in the Examples framework. Currently all our writers have a separate truth matching implementation with some amount of common code. I propose to move the truth matching to its own algorithm and then use the output as an input for the writers. This also allows for simple truth matching in case of truth tracking to be specialized and for a common mechanism for how we infer the truth information. E.g. some algorithms right now expect particles and tracks in a 1:1 correspondence in the input containers which is not achievable with CKF tracking. blocked by - acts-project#2964
- Loading branch information
Showing
56 changed files
with
1,344 additions
and
911 deletions.
There are no files selected for viewing
Binary file modified
BIN
-3.13 KB
(94%)
CI/physmon/reference/performance_amvf_gridseeder_seeded_hist.root
Binary file not shown.
Binary file modified
BIN
-3.08 KB
(94%)
CI/physmon/reference/performance_amvf_gridseeder_ttbar_hist.root
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file modified
BIN
-3.33 KB
(93%)
CI/physmon/reference/performance_amvf_truth_estimated_hist.root
Binary file not shown.
Binary file modified
BIN
-3.24 KB
(93%)
CI/physmon/reference/performance_amvf_truth_smeared_hist.root
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file modified
BIN
-2.83 KB
(94%)
CI/physmon/reference/performance_ivf_truth_estimated_hist.root
Binary file not shown.
Binary file modified
BIN
-2.74 KB
(95%)
CI/physmon/reference/performance_ivf_truth_smeared_hist.root
Binary file not shown.
Binary file not shown.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
151 changes: 151 additions & 0 deletions
151
Examples/Algorithms/TruthTracking/ActsExamples/TruthTracking/ProtoTrackTruthMatcher.cpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,151 @@ | ||
// This file is part of the Acts project. | ||
// | ||
// Copyright (C) 2024 CERN for the benefit of the Acts project | ||
// | ||
// This Source Code Form is subject to the terms of the Mozilla Public | ||
// License, v. 2.0. If a copy of the MPL was not distributed with this | ||
// file, You can obtain one at http://mozilla.org/MPL/2.0/. | ||
|
||
#include "ActsExamples/TruthTracking/ProtoTrackTruthMatcher.hpp" | ||
|
||
#include "Acts/Utilities/Enumerate.hpp" | ||
#include "ActsExamples/EventData/SimParticle.hpp" | ||
#include "ActsExamples/Validation/TrackClassification.hpp" | ||
|
||
#include <map> | ||
#include <optional> | ||
#include <stdexcept> | ||
#include <vector> | ||
|
||
namespace ActsExamples { | ||
|
||
ProtoTrackTruthMatcher::ProtoTrackTruthMatcher(const Config& config, | ||
Acts::Logging::Level level) | ||
: IAlgorithm("ProtoTrackTruthMatcher", level), m_cfg(config) { | ||
if (m_cfg.inputProtoTracks.empty()) { | ||
throw std::invalid_argument("Missing input proto tracks"); | ||
} | ||
if (m_cfg.inputParticles.empty()) { | ||
throw std::invalid_argument("Missing input particles"); | ||
} | ||
if (m_cfg.inputMeasurementParticlesMap.empty()) { | ||
throw std::invalid_argument("Missing input measurement particles map"); | ||
} | ||
if (m_cfg.outputProtoTrackParticleMatching.empty()) { | ||
throw std::invalid_argument( | ||
"Missing output proto track particles matching"); | ||
} | ||
if (m_cfg.outputParticleProtoTrackMatching.empty()) { | ||
throw std::invalid_argument("Missing output particle proto track matching"); | ||
} | ||
|
||
m_inputProtoTracks.initialize(m_cfg.inputProtoTracks); | ||
m_inputParticles.initialize(m_cfg.inputParticles); | ||
m_inputMeasurementParticlesMap.initialize(m_cfg.inputMeasurementParticlesMap); | ||
m_outputProtoTrackParticleMatching.initialize( | ||
m_cfg.outputProtoTrackParticleMatching); | ||
m_outputParticleProtoTrackMatching.initialize( | ||
m_cfg.outputParticleProtoTrackMatching); | ||
} | ||
|
||
ActsExamples::ProcessCode ProtoTrackTruthMatcher::execute( | ||
const ActsExamples::AlgorithmContext& ctx) const { | ||
// Read input proto tracks | ||
const auto& protoTracks = m_inputProtoTracks(ctx); | ||
|
||
// Read truth input collections | ||
const auto& particles = m_inputParticles(ctx); | ||
const auto& hitParticlesMap = m_inputMeasurementParticlesMap(ctx); | ||
|
||
ProtoTrackParticleMatching protoTrackParticleMatching; | ||
ParticleProtoTrackMatching particleProtoTrackMatching; | ||
|
||
// TODO this may be computed in a separate algorithm | ||
// TODO can we wire this through? | ||
std::map<SimBarcode, std::size_t> particleTruthHitCount; | ||
for (const auto& [_, pid] : hitParticlesMap) { | ||
particleTruthHitCount[pid]++; | ||
} | ||
|
||
// For each particle within a track, how many hits did it contribute | ||
std::vector<ParticleHitCount> particleHitCounts; | ||
|
||
for (const auto& [index, protoTrack] : Acts::enumerate(protoTracks)) { | ||
// Get the majority truth particle to this track | ||
identifyContributingParticles(hitParticlesMap, protoTrack, | ||
particleHitCounts); | ||
if (particleHitCounts.empty()) { | ||
ACTS_DEBUG("No truth particle associated with this proto track " | ||
<< index); | ||
continue; | ||
} | ||
|
||
// Get the majority particleId and majority particle counts | ||
// Note that the majority particle might not be in the truth seeds | ||
// collection | ||
ActsFatras::Barcode majorityParticleId = | ||
particleHitCounts.front().particleId; | ||
std::size_t nMajorityHits = particleHitCounts.front().hitCount; | ||
|
||
if (particles.find(majorityParticleId) == particles.end()) { | ||
ACTS_DEBUG( | ||
"The majority particle is not in the input particle collection, " | ||
"majorityParticleId = " | ||
<< majorityParticleId); | ||
continue; | ||
} | ||
|
||
// Check if the trajectory is matched with truth. | ||
// If not, it will be classified as 'fake' | ||
const bool recoMatched = | ||
static_cast<double>(nMajorityHits) / protoTrack.size() >= | ||
m_cfg.matchingRatio; | ||
const bool truthMatched = | ||
static_cast<double>(nMajorityHits) / | ||
particleTruthHitCount.at(majorityParticleId) >= | ||
m_cfg.matchingRatio; | ||
|
||
if ((!m_cfg.doubleMatching && recoMatched) || | ||
(m_cfg.doubleMatching && recoMatched && truthMatched)) { | ||
auto& trackParticleMatch = protoTrackParticleMatching[index] = { | ||
TrackMatchClassification::Matched, majorityParticleId, | ||
particleHitCounts}; | ||
|
||
auto& particleTrackMatch = particleProtoTrackMatching[majorityParticleId]; | ||
if (!particleTrackMatch.track) { | ||
particleTrackMatch.track = index; | ||
} else { | ||
// we already have a track associated with this particle and have to | ||
// resolve the ambiguity. | ||
// we will use the track with more hits and smaller chi2 | ||
const auto& otherProtoTrack = | ||
protoTracks.at(particleTrackMatch.track.value()); | ||
if (otherProtoTrack.size() < protoTrack.size()) { | ||
protoTrackParticleMatching[particleTrackMatch.track.value()] | ||
.classification = TrackMatchClassification::Duplicate; | ||
particleTrackMatch.track = index; | ||
} else { | ||
trackParticleMatch.classification = | ||
TrackMatchClassification::Duplicate; | ||
} | ||
|
||
++particleTrackMatch.duplicates; | ||
} | ||
} else { | ||
protoTrackParticleMatching[index] = {TrackMatchClassification::Fake, | ||
std::nullopt, particleHitCounts}; | ||
|
||
auto& particleTrackMatch = particleProtoTrackMatching[majorityParticleId]; | ||
++particleTrackMatch.fakes; | ||
} | ||
} | ||
|
||
m_outputProtoTrackParticleMatching(ctx, | ||
std::move(protoTrackParticleMatching)); | ||
m_outputParticleProtoTrackMatching(ctx, | ||
std::move(particleProtoTrackMatching)); | ||
|
||
return ProcessCode::SUCCESS; | ||
} | ||
|
||
} // namespace ActsExamples |
68 changes: 68 additions & 0 deletions
68
Examples/Algorithms/TruthTracking/ActsExamples/TruthTracking/ProtoTrackTruthMatcher.hpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,68 @@ | ||
// This file is part of the Acts project. | ||
// | ||
// Copyright (C) 2024 CERN for the benefit of the Acts project | ||
// | ||
// This Source Code Form is subject to the terms of the Mozilla Public | ||
// License, v. 2.0. If a copy of the MPL was not distributed with this | ||
// file, You can obtain one at http://mozilla.org/MPL/2.0/. | ||
|
||
#pragma once | ||
|
||
#include "Acts/Utilities/Logger.hpp" | ||
#include "ActsExamples/EventData/ProtoTrack.hpp" | ||
#include "ActsExamples/EventData/SimHit.hpp" | ||
#include "ActsExamples/EventData/SimParticle.hpp" | ||
#include "ActsExamples/EventData/TruthMatching.hpp" | ||
#include "ActsExamples/Framework/DataHandle.hpp" | ||
#include "ActsExamples/Framework/IAlgorithm.hpp" | ||
#include "ActsExamples/Framework/ProcessCode.hpp" | ||
|
||
#include <string> | ||
|
||
namespace ActsExamples { | ||
|
||
struct AlgorithmContext; | ||
|
||
/// Matches proto track to truth particles and vice versa | ||
class ProtoTrackTruthMatcher final : public IAlgorithm { | ||
public: | ||
struct Config { | ||
/// Input proto tracks collection | ||
std::string inputProtoTracks; | ||
/// Input particles collection. | ||
std::string inputParticles; | ||
/// Input hit-particles map collection. | ||
std::string inputMeasurementParticlesMap; | ||
/// Output proto track-particle matching. | ||
std::string outputProtoTrackParticleMatching; | ||
/// Output particle-proto track matching. | ||
std::string outputParticleProtoTrackMatching; | ||
|
||
/// Matching ratio for track to particle matching | ||
double matchingRatio = 0.5; | ||
/// Whether to use double matching (track to particle and particle to track) | ||
bool doubleMatching = false; | ||
}; | ||
|
||
ProtoTrackTruthMatcher(const Config& config, Acts::Logging::Level level); | ||
|
||
ProcessCode execute(const AlgorithmContext& ctx) const final; | ||
|
||
/// Get readonly access to the config parameters | ||
const Config& config() const { return m_cfg; } | ||
|
||
private: | ||
Config m_cfg; | ||
|
||
ReadDataHandle<ProtoTrackContainer> m_inputProtoTracks{this, | ||
"InputProtoTracks"}; | ||
ReadDataHandle<SimParticleContainer> m_inputParticles{this, "InputParticles"}; | ||
ReadDataHandle<HitParticlesMap> m_inputMeasurementParticlesMap{ | ||
this, "InputMeasurementParticlesMap"}; | ||
WriteDataHandle<TrackParticleMatching> m_outputProtoTrackParticleMatching{ | ||
this, "OutputProtoTrackParticleMatching"}; | ||
WriteDataHandle<ParticleTrackMatching> m_outputParticleProtoTrackMatching{ | ||
this, "OutputParticleProtoTrackMatching"}; | ||
}; | ||
|
||
} // namespace ActsExamples |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.