CMS 3D CMS Logo

 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Pages
ThinnedAssociationsHelper.cc
Go to the documentation of this file.
4 #include <algorithm>
5 
6 namespace edm {
7 
9 
11  BranchID const& association,
12  BranchID const& thinned) :
13  parent_(parent),
14  association_(association),
15  thinned_(thinned) {
16  }
17 
19  }
20 
21  std::vector<ThinnedAssociationBranches>::const_iterator
23  return vThinnedAssociationBranches_.begin();
24  }
25 
26  std::vector<ThinnedAssociationBranches>::const_iterator
28  return vThinnedAssociationBranches_.end();
29  }
30 
31  std::vector<ThinnedAssociationBranches>::const_iterator
34  return std::lower_bound(vThinnedAssociationBranches_.begin(), vThinnedAssociationBranches_.end(), target,
35  [](ThinnedAssociationBranches const& x,
37  { return x.parent() < y.parent(); });
38  }
39 
40  std::vector<ThinnedAssociationBranches>::const_iterator
43  return std::upper_bound(vThinnedAssociationBranches_.begin(), vThinnedAssociationBranches_.end(), target,
44  [](ThinnedAssociationBranches const& x,
46  { return x.parent() < y.parent(); });
47  }
48 
51  [](ThinnedAssociationBranches const& x,
53  { return x.parent() < y.parent() ? true : y.parent() < x.parent() ? false : x.association() < y.association(); });
54  }
55 
57  BranchID const& association,
58  BranchID const& thinned) {
59  vThinnedAssociationBranches_.push_back(ThinnedAssociationBranches(parent, association, thinned));
60  }
61 
63  vThinnedAssociationBranches_.push_back(branches);
64  }
65 
66  std::vector<std::pair<BranchID, ThinnedAssociationBranches const*> >
68  std::vector<std::pair<BranchID, ThinnedAssociationBranches const*> > temp;
69  for(auto const& item : vThinnedAssociationBranches_) {
70  temp.push_back(std::make_pair(item.association(), &item));
71  }
72  std::sort(temp.begin(), temp.end(), [](std::pair<BranchID, ThinnedAssociationBranches const*> const& x,
73  std::pair<BranchID, ThinnedAssociationBranches const*> const& y)
74  { return x.first < y.first; });
75  return temp;
76  }
77 
78  void
80  selectAssociationProducts(std::vector<BranchDescription const*> const& associationDescriptions,
81  std::set<BranchID> const& keptProductsInEvent,
82  std::map<BranchID, bool>& keepAssociation) const {
83 
84  keepAssociation.clear();
85  // Copy the elements in vThinnedAssociationBranches_ into a vector sorted on
86  // the association BranchID so we can do searches on that BranchID faster.
87  std::vector<std::pair<BranchID, ThinnedAssociationBranches const*> > assocToBranches =
89 
90  for(auto association : associationDescriptions) {
91  if(association->isAlias()) { // There is no reason to configure an association product with an EDAlias (ignore and drop them if they exist)
92  keepAssociation.insert(std::make_pair(association->branchID(), false));
93  } else {
94  std::set<BranchID> branchesInRecursion;
95  shouldKeepAssociation(association->branchID(),
96  assocToBranches,
97  branchesInRecursion,
98  keptProductsInEvent,
99  keepAssociation);
100  }
101  }
102  }
103 
105  BranchID const& association,
106  std::vector<std::pair<BranchID, ThinnedAssociationBranches const*> > const& associationToBranches,
107  std::set<BranchID>& branchesInRecursion,
108  std::set<BranchID> const& keptProductsInEvent,
109  std::map<BranchID, bool>& keepAssociation) const {
110 
111  // If we already decided to keep or drop this one, then
112  // return the same decision.
113  auto decision = keepAssociation.find(association);
114  if(decision != keepAssociation.end()) {
115  return decision->second;
116  }
117 
118  // Be careful not to fall into an infinite loop because
119  // of a circular recursion.
120  if(!branchesInRecursion.insert(association).second) {
121  return false;
122  }
123 
124  // If the thinned collection is being kept then keep the association
125  auto branches = std::lower_bound(associationToBranches.begin(), associationToBranches.end(),
126  std::make_pair(association, static_cast<ThinnedAssociationBranches const*>(nullptr)),
127  [](std::pair<BranchID, ThinnedAssociationBranches const*> const& x,
128  std::pair<BranchID, ThinnedAssociationBranches const*> const& y)
129  { return x.first < y.first; });
130  // This should never happen
131  if(branches == associationToBranches.end() || branches->first != association) {
132  throw edm::Exception(errors::LogicError, "ThinnedAssociationHelper::shouldKeepAssociation could not find branches information, contact Framework developers");
133  }
134  BranchID const& thinnedCollection = branches->second->thinned();
135  if(keptProductsInEvent.find(thinnedCollection) != keptProductsInEvent.end()) {
136  keepAssociation.insert(std::make_pair(association , true));
137  return true;
138  }
139  // otherwise loop over any associations where the thinned collection
140  // is also a parent collection and recursively examine those to see
141  // if their thinned collections are being kept.
142  auto iterEnd = parentEnd(thinnedCollection);
143  for(auto match = parentBegin(thinnedCollection); match != iterEnd; ++match) {
144  if(shouldKeepAssociation(match->association(),
146  branchesInRecursion,
147  keptProductsInEvent,
148  keepAssociation)) {
149  keepAssociation.insert(std::make_pair(association , true));
150  return true;
151  }
152  }
153  // drop the association
154  keepAssociation.insert(std::make_pair(association , false));
155  return false;
156  }
157 
159  bool foundMatch = false;
160  for(auto entry = parentBegin(input.parent()), iEnd = parentEnd(input.parent());
161  entry != iEnd; ++entry) {
162  if(entry->association() == input.association() &&
163  entry->thinned() == input.thinned()) {
164  foundMatch = true;
165  break;
166  }
167  }
168  if(!foundMatch) {
169  throw edm::Exception(errors::MismatchedInputFiles, "ThinnedAssociationHelper::requireMatch, Illegal attempt to merge files with different ThinnedAssociations");
170  }
171  }
172 
174  std::vector<BranchID> const& associationsFromSecondary) {
175  if(!isSecondaryFile) {
176  if(vThinnedAssociationBranches_.empty()) {
178  return;
179  }
180  std::vector<ThinnedAssociationBranches> const& inputData = helper.data();
181  for (auto const& inputEntry : inputData) {
182  requireMatch(inputEntry);
183  }
184  } else { // Input is from a secondary file
185 
186  if(associationsFromSecondary.empty()) return;
187 
188  std::vector<std::pair<BranchID, ThinnedAssociationBranches const*> > assocToBranches = helper.associationToBranches();
189 
190  for(BranchID const& association : associationsFromSecondary) {
191 
192  auto branches = std::lower_bound(assocToBranches.begin(), assocToBranches.end(),
193  std::make_pair(association, static_cast<ThinnedAssociationBranches const*>(nullptr)),
194  [](std::pair<BranchID, ThinnedAssociationBranches const*> const& x,
195  std::pair<BranchID, ThinnedAssociationBranches const*> const& y)
196  { return x.first < y.first; });
197  // This should never happen
198  if(branches == assocToBranches.end() || branches->first != association) {
200  "ThinnedAssociationHelper::initAssociationsFromSecondary could not find branches information, contact Framework developers");
201  }
202  requireMatch(*(branches->second));
203  }
204  }
205  }
206 
207  void
209  std::map<BranchID, bool> const& keepAssociation,
210  std::map<BranchID::value_type, BranchID::value_type> const& droppedBranchIDToKeptBranchID) {
211  clear();
212  for(auto const& associationBranches : parentThinnedAssociationsHelper.data()) {
213  auto keep = keepAssociation.find(associationBranches.association());
214  if(keep != keepAssociation.end() && keep->second) {
215  BranchID parent = associationBranches.parent();
216  auto iter = droppedBranchIDToKeptBranchID.find(parent.id());
217  if(iter != droppedBranchIDToKeptBranchID.end()) {
218  parent = BranchID(iter->second);
219  }
220  BranchID thinned = associationBranches.thinned();
221  iter = droppedBranchIDToKeptBranchID.find(thinned.id());
222  if(iter != droppedBranchIDToKeptBranchID.end()) {
223  thinned = BranchID(iter->second);
224  }
225  addAssociation(parent, associationBranches.association(), thinned);
226  }
227  }
228  sort();
229  }
230 
231  void
232  ThinnedAssociationsHelper::initAssociationsFromSecondary(std::vector<BranchID> const& associationsFromSecondary,
233  ThinnedAssociationsHelper const& fileAssociationsHelper) {
234  if(associationsFromSecondary.empty()) return;
235 
236  std::vector<std::pair<BranchID, ThinnedAssociationBranches const*> > assocToBranches =
237  fileAssociationsHelper.associationToBranches();
238 
239  for(BranchID const& association : associationsFromSecondary) {
240 
241  auto branches = std::lower_bound(assocToBranches.begin(), assocToBranches.end(),
242  std::make_pair(association, static_cast<ThinnedAssociationBranches const*>(nullptr)),
243  [](std::pair<BranchID, ThinnedAssociationBranches const*> const& x,
244  std::pair<BranchID, ThinnedAssociationBranches const*> const& y)
245  { return x.first < y.first; });
246  // This should never happen
247  if(branches == assocToBranches.end() || branches->first != association) {
249  "ThinnedAssociationHelper::initAssociationsFromSecondary could not find branches information, contact Framework developers");
250  }
251  addAssociation(*(branches->second));
252  }
253  sort();
254  }
255 }
void initAssociationsFromSecondary(std::vector< BranchID > const &, ThinnedAssociationsHelper const &)
std::vector< ThinnedAssociationBranches > const & data() const
list parent
Definition: dbtoconf.py:74
void addAssociation(BranchID const &, BranchID const &, BranchID const &)
void requireMatch(ThinnedAssociationBranches const &input) const
std::vector< std::pair< BranchID, ThinnedAssociationBranches const * > > associationToBranches() const
void updateFromInput(ThinnedAssociationsHelper const &, bool isSecondaryFile, std::vector< BranchID > const &associationsFromSecondary)
unsigned int id() const
Definition: BranchID.h:23
std::vector< ThinnedAssociationBranches >::const_iterator end() const
static std::string const input
Definition: EdmProvDump.cc:44
std::vector< ThinnedAssociationBranches >::const_iterator parentEnd(BranchID const &) const
void selectAssociationProducts(std::vector< BranchDescription const * > const &associationDescriptions, std::set< BranchID > const &keptProductsInEvent, std::map< BranchID, bool > &keepAssociation) const
const int keep
std::vector< ThinnedAssociationBranches > vThinnedAssociationBranches_
std::vector< ThinnedAssociationBranches >::const_iterator begin() const
std::vector< ThinnedAssociationBranches >::const_iterator parentBegin(BranchID const &) const
void updateFromParentProcess(ThinnedAssociationsHelper const &parentThinnedAssociationsHelper, std::map< BranchID, bool > const &keepAssociation, std::map< BranchID::value_type, BranchID::value_type > const &droppedBranchIDToKeptBranchID)
bool shouldKeepAssociation(BranchID const &association, std::vector< std::pair< BranchID, ThinnedAssociationBranches const * > > const &associationToBranches, std::set< BranchID > &branchesInRecursion, std::set< BranchID > const &keptProductsInEvent, std::map< BranchID, bool > &keepAssociation) const
Definition: DDAxes.h:10
tuple inputData
Definition: idDealer.py:72
std::string match(BranchDescription const &a, BranchDescription const &b, std::string const &fileName)