ProteoWizard
IterationListenerTest.cpp
Go to the documentation of this file.
1 //
2 // $Id: IterationListenerTest.cpp 4129 2012-11-20 00:05:37Z chambm $
3 //
4 //
5 // Original author: Darren Kessner <darren@proteowizard.org>
6 //
7 // Copyright 2008 Spielberg Family Center for Applied Proteomics
8 // Cedars Sinai Medical Center, Los Angeles, California 90048
9 //
10 // Licensed under the Apache License, Version 2.0 (the "License");
11 // you may not use this file except in compliance with the License.
12 // You may obtain a copy of the License at
13 //
14 // http://www.apache.org/licenses/LICENSE-2.0
15 //
16 // Unless required by applicable law or agreed to in writing, software
17 // distributed under the License is distributed on an "AS IS" BASIS,
18 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19 // See the License for the specific language governing permissions and
20 // limitations under the License.
21 //
22 
23 
24 #include "Std.hpp"
25 #include "IterationListener.hpp"
27 #include <cstring>
28 #include <ctime>
29 
30 
31 using namespace pwiz::util;
32 
33 
34 ostream* os_ = 0;
35 
36 
38 {
39  public:
40 
41  TestListener(const string& name)
42  : name_(name), count_(0)
43  {}
44 
45  virtual Status update(const UpdateMessage& updateMessage)
46  {
47  if (os_) *os_ << "[" << name_ << "] " << updateMessage.iterationIndex << "/"
48  << updateMessage.iterationCount << endl;
49  count_++;
50  return Status_Ok;
51  }
52 
53  size_t count() const {return count_;}
54 
55  private:
56  string name_;
57  size_t count_;
58 };
59 
60 
62 {
63  public:
64 
65  CancelListener(size_t cancelIndex)
66  : cancelIndex_(cancelIndex)
67  {}
68 
69  virtual Status update(const UpdateMessage& updateMessage)
70  {
71  if (os_) *os_ << "[cancel] " << updateMessage.iterationIndex << "/"
72  << updateMessage.iterationCount << endl;
73 
74  return updateMessage.iterationIndex==cancelIndex_ ? Status_Cancel : Status_Ok;
75  }
76 
77  private:
78  size_t cancelIndex_;
79 };
80 
81 
82 // null deallactor to create shared_ptrs that do not delete when reset
84 {
85  // do nothing
86 }
87 
88 void test()
89 {
90  if (os_) *os_ << "test()\n";
91 
93 
94  TestListener test3("test3");
95  TestListener test4("test4");
96  TestListener test5("test5");
97  TestListener test6("test6");
98 
99  registry.addListener(IterationListenerPtr(&test3, nullDeallocate), 3);
100  registry.addListener(IterationListenerPtr(&test4, nullDeallocate), 4);
101  registry.addListener(IterationListenerPtr(&test5, nullDeallocate), 5);
102  registry.addListener(IterationListenerPtr(&test6, nullDeallocate), 6);
103 
104  size_t iterationCount = 24;
105  for (size_t i=0; i<iterationCount; i++)
106  registry.broadcastUpdateMessage(IterationListener::UpdateMessage(i, iterationCount));
107 
108  // validate
109 
110  unit_assert(test3.count() == 9); // 0 2 5 8 11 14 17 20 23
111  unit_assert(test4.count() == 7);
112  unit_assert(test5.count() == 6);
113  unit_assert(test6.count() == 5);
114 
115  if (os_) *os_ << endl;
116 }
117 
118 
120 {
121  if (os_) *os_ << "testCancel()\n";
122 
123  IterationListenerRegistry registry;
124 
125  CancelListener cancelListener(12);
126  TestListener test3("test3");
127  TestListener test4("test4");
128  TestListener test6("test6");
129 
130  registry.addListener(IterationListenerPtr(&cancelListener, nullDeallocate), 1);
131  registry.addListener(IterationListenerPtr(&test3, nullDeallocate), 3);
132  registry.addListener(IterationListenerPtr(&test4, nullDeallocate), 4);
133  registry.addListener(IterationListenerPtr(&test6, nullDeallocate), 5);
134 
135  // typical use of IterationListenerRegistry, with proper Status_Cancel handling
136 
137  bool canceled = false;
138 
139  size_t iterationCount = 24;
140  for (size_t i=0; i<iterationCount; i++)
141  {
142  IterationListener::Status status =
143  registry.broadcastUpdateMessage(IterationListener::UpdateMessage(i, iterationCount));
144 
145  // handle Status_Cancel
146  if (status == IterationListener::Status_Cancel)
147  {
148  canceled = true;
149  break;
150  }
151  }
152 
153  // implementations should send a final update on completion of the iteration
154 
155  if (!canceled)
156  registry.broadcastUpdateMessage(IterationListener::UpdateMessage(iterationCount, iterationCount));
157 
158  // validate
159 
160  unit_assert(test3.count() == 5);
161  unit_assert(test4.count() == 4);
162  unit_assert(test6.count() == 3);
163 
164  if (os_) *os_ << endl;
165 }
166 
167 
169 {
170  public:
171 
172  virtual Status update(const UpdateMessage& updateMessage)
173  {
174  throw runtime_error("bad");
175  }
176 };
177 
178 
180 {
181  if (os_) *os_ << "testRemove()\n";
182 
183  IterationListenerRegistry registry;
184 
185  BadListener bad;
186  TestListener test3("test3");
187  TestListener test4("test4");
188 
189  IterationListenerPtr badPtr(&bad, nullDeallocate);
190 
191  registry.addListener(IterationListenerPtr(&test3, nullDeallocate), 3);
192  registry.addListener(badPtr, 1);
193  registry.addListener(IterationListenerPtr(&test4, nullDeallocate), 4);
194 
195  // sanity check -- verify that broadcast throws if BadListener is in the registry
196 
197  bool caught = false;
198 
199  try
200  {
202  }
203  catch (exception& e)
204  {
205  if (e.what() == string("bad")) caught = true;
206  }
207 
208  unit_assert(caught);
209 
210  // remove BadListener -- broadcast will throw if not removed properly
211 
212  registry.removeListener(badPtr);
214 
215  if (os_) *os_ << endl;
216 }
217 
218 
219 void testTime()
220 {
221  if (os_) *os_ << "testTime()\n";
222 
223  IterationListenerRegistry registry;
224 
225  TestListener test_iteration("test_iteration");
226  TestListener test_time("test_time");
227 
228  registry.addListener(IterationListenerPtr(&test_iteration, nullDeallocate), 1000000);
229  registry.addListenerWithTimer(IterationListenerPtr(&test_time, nullDeallocate), 1.0);
230 
231  time_t start;
232  time(&start);
233 
234  const double iterationDuration = 5.0;
235  for (int i=0; ; i++)
236  {
237  time_t now;
238  time(&now);
239  if (difftime(now, start) > iterationDuration) break;
240 
242  }
243 
244  if (os_) *os_ << endl;
245 }
246 
247 
248 int main(int argc, char* argv[])
249 {
250  TEST_PROLOG(argc, argv)
251 
252  try
253  {
254  if (argc>1 && !strcmp(argv[1],"-v")) os_ = &cout;
255  test();
256  testCancel();
257  testRemove();
258  testTime();
259  }
260  catch (exception& e)
261  {
262  TEST_FAILED(e.what())
263  }
264  catch (...)
265  {
266  TEST_FAILED("Caught unknown exception.")
267  }
268 
270 }
271 
272