blob: 3779b0d18f0d06d9eba4af0fbfc6743b641903b6 [file] [log] [blame]
David Reissea2cba82009-03-30 21:35:00 +00001/*
2 * Licensed to the Apache Software Foundation (ASF) under one
3 * or more contributor license agreements. See the NOTICE file
4 * distributed with this work for additional information
5 * regarding copyright ownership. The ASF licenses this file
6 * to you under the Apache License, Version 2.0 (the
7 * "License"); you may not use this file except in compliance
8 * with the License. You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing,
13 * software distributed under the License is distributed on an
14 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 * KIND, either express or implied. See the License for the
16 * specific language governing permissions and limitations
17 * under the License.
18 */
Mark Slee9f0c6512007-02-28 23:58:26 +000019
Roger Meier49ff8b12012-04-13 09:12:31 +000020#include <thrift/concurrency/TimerManager.h>
21#include <thrift/concurrency/PlatformThreadFactory.h>
22#include <thrift/concurrency/Monitor.h>
23#include <thrift/concurrency/Util.h>
Marc Slemko8a40a762006-07-19 17:46:50 +000024
25#include <assert.h>
26#include <iostream>
27
Konrad Grochowski16a23a62014-11-13 15:33:38 +010028namespace apache {
29namespace thrift {
30namespace concurrency {
31namespace test {
Marc Slemko8a40a762006-07-19 17:46:50 +000032
T Jake Lucianib5e62212009-01-31 22:36:20 +000033using namespace apache::thrift::concurrency;
Marc Slemko8a40a762006-07-19 17:46:50 +000034
Marc Slemko8a40a762006-07-19 17:46:50 +000035class TimerManagerTests {
36
Konrad Grochowski16a23a62014-11-13 15:33:38 +010037public:
38 class Task : public Runnable {
39 public:
40 Task(Monitor& monitor, int64_t timeout)
41 : _timeout(timeout),
42 _startTime(Util::currentTime()),
James E. King, III36200902016-10-05 14:47:18 -040043 _endTime(0),
Konrad Grochowski16a23a62014-11-13 15:33:38 +010044 _monitor(monitor),
45 _success(false),
46 _done(false) {}
Marc Slemko8a40a762006-07-19 17:46:50 +000047
Mark Sleef5f2be42006-09-05 21:05:31 +000048 ~Task() { std::cerr << this << std::endl; }
Marc Slemko6f038a72006-08-03 18:58:09 +000049
Marc Slemko8a40a762006-07-19 17:46:50 +000050 void run() {
51
Marc Slemko9f27a4e2006-07-19 20:02:22 +000052 _endTime = Util::currentTime();
James E. King, IIIdf899132016-11-12 15:16:30 -050053 _success = (_endTime - _startTime) >= _timeout;
Marc Slemko6f038a72006-08-03 18:58:09 +000054
Konrad Grochowski16a23a62014-11-13 15:33:38 +010055 {
56 Synchronized s(_monitor);
James E. King, IIIdf899132016-11-12 15:16:30 -050057 _done = true;
David Reiss96d23882007-07-26 21:10:32 +000058 _monitor.notifyAll();
Marc Slemko8a40a762006-07-19 17:46:50 +000059 }
David Reiss0c90f6f2008-02-06 22:18:40 +000060 }
Marc Slemko9f27a4e2006-07-19 20:02:22 +000061
Mark Slee9b82d272007-05-23 05:16:07 +000062 int64_t _timeout;
63 int64_t _startTime;
64 int64_t _endTime;
Marc Slemko8a40a762006-07-19 17:46:50 +000065 Monitor& _monitor;
Marc Slemko9f27a4e2006-07-19 20:02:22 +000066 bool _success;
Marc Slemko8a40a762006-07-19 17:46:50 +000067 bool _done;
68 };
69
Mark Sleef5f2be42006-09-05 21:05:31 +000070 /**
71 * This test creates two tasks and waits for the first to expire within 10%
72 * of the expected expiration time. It then verifies that the timer manager
73 * properly clean up itself and the remaining orphaned timeout task when the
74 * manager goes out of scope and its destructor is called.
75 */
Konrad Grochowski16a23a62014-11-13 15:33:38 +010076 bool test00(int64_t timeout = 1000LL) {
Marc Slemko8a40a762006-07-19 17:46:50 +000077
Konrad Grochowski16a23a62014-11-13 15:33:38 +010078 shared_ptr<TimerManagerTests::Task> orphanTask
79 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, 10 * timeout));
Marc Slemko8a40a762006-07-19 17:46:50 +000080
Marc Slemko9f27a4e2006-07-19 20:02:22 +000081 {
Marc Slemko8a40a762006-07-19 17:46:50 +000082
Marc Slemko9f27a4e2006-07-19 20:02:22 +000083 TimerManager timerManager;
David Reiss0c90f6f2008-02-06 22:18:40 +000084
Roger Meier3faaedf2011-10-02 10:51:45 +000085 timerManager.threadFactory(shared_ptr<PlatformThreadFactory>(new PlatformThreadFactory()));
David Reiss0c90f6f2008-02-06 22:18:40 +000086
Marc Slemko9f27a4e2006-07-19 20:02:22 +000087 timerManager.start();
David Reiss0c90f6f2008-02-06 22:18:40 +000088
Marc Slemko9f27a4e2006-07-19 20:02:22 +000089 assert(timerManager.state() == TimerManager::STARTED);
Marc Slemko8a40a762006-07-19 17:46:50 +000090
David Reiss52687eb2009-06-04 00:32:57 +000091 // Don't create task yet, because its constructor sets the expected completion time, and we
92 // need to delay between inserting the two tasks into the run queue.
93 shared_ptr<TimerManagerTests::Task> task;
Marc Slemko8a40a762006-07-19 17:46:50 +000094
Mark Sleef5f2be42006-09-05 21:05:31 +000095 {
96 Synchronized s(_monitor);
Marc Slemko8a40a762006-07-19 17:46:50 +000097
David Reiss96d23882007-07-26 21:10:32 +000098 timerManager.add(orphanTask, 10 * timeout);
Marc Slemko8a40a762006-07-19 17:46:50 +000099
David Reiss52687eb2009-06-04 00:32:57 +0000100 try {
101 // Wait for 1 second in order to give timerManager a chance to start sleeping in response
102 // to adding orphanTask. We need to do this so we can verify that adding the second task
103 // kicks the dispatcher out of the current wait and starts the new 1 second wait.
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100104 _monitor.wait(1000);
105 assert(
106 0 == "ERROR: This wait should time out. TimerManager dispatcher may have a problem.");
ben-craigfae08e72015-07-15 11:34:47 -0500107 } catch (TimedOutException&) {
David Reiss52687eb2009-06-04 00:32:57 +0000108 }
109
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100110 task.reset(new TimerManagerTests::Task(_monitor, timeout));
David Reiss52687eb2009-06-04 00:32:57 +0000111
David Reiss96d23882007-07-26 21:10:32 +0000112 timerManager.add(task, timeout);
Marc Slemko9f27a4e2006-07-19 20:02:22 +0000113
David Reiss96d23882007-07-26 21:10:32 +0000114 _monitor.wait();
Marc Slemko9f27a4e2006-07-19 20:02:22 +0000115 }
116
117 assert(task->_done);
118
Marc Slemko9f27a4e2006-07-19 20:02:22 +0000119 std::cout << "\t\t\t" << (task->_success ? "Success" : "Failure") << "!" << std::endl;
Marc Slemko8a40a762006-07-19 17:46:50 +0000120 }
121
Marc Slemko9f27a4e2006-07-19 20:02:22 +0000122 // timerManager.stop(); This is where it happens via destructor
Marc Slemko8a40a762006-07-19 17:46:50 +0000123
Marc Slemko9f27a4e2006-07-19 20:02:22 +0000124 assert(!orphanTask->_done);
Marc Slemko8a40a762006-07-19 17:46:50 +0000125
Marc Slemko8a40a762006-07-19 17:46:50 +0000126 return true;
127 }
128
Francois Ferrandcc2d5582017-08-25 09:01:26 +0200129 /**
130 * This test creates two tasks, removes the first one then waits for the second one. It then
131 * verifies that the timer manager properly clean up itself and the remaining orphaned timeout
132 * task when the manager goes out of scope and its destructor is called.
133 */
134 bool test01(int64_t timeout = 1000LL) {
135 TimerManager timerManager;
136 timerManager.threadFactory(shared_ptr<PlatformThreadFactory>(new PlatformThreadFactory()));
137 timerManager.start();
138 assert(timerManager.state() == TimerManager::STARTED);
139
140 Synchronized s(_monitor);
141
142 // Setup the two tasks
143 shared_ptr<TimerManagerTests::Task> taskToRemove
144 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout / 2));
145 timerManager.add(taskToRemove, taskToRemove->_timeout);
146
147 shared_ptr<TimerManagerTests::Task> task
148 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout));
149 timerManager.add(task, task->_timeout);
150
151 // Remove one task and wait until the other has completed
152 timerManager.remove(taskToRemove);
153 _monitor.wait(timeout * 2);
154
155 assert(!taskToRemove->_done);
156 assert(task->_done);
157
158 return true;
159 }
160
161 /**
162 * This test creates two tasks with the same callback and another one, then removes the two
163 * duplicated then waits for the last one. It then verifies that the timer manager properly
164 * clean up itself and the remaining orphaned timeout task when the manager goes out of scope
165 * and its destructor is called.
166 */
167 bool test02(int64_t timeout = 1000LL) {
168 TimerManager timerManager;
169 timerManager.threadFactory(shared_ptr<PlatformThreadFactory>(new PlatformThreadFactory()));
170 timerManager.start();
171 assert(timerManager.state() == TimerManager::STARTED);
172
173 Synchronized s(_monitor);
174
175 // Setup the one tasks and add it twice
176 shared_ptr<TimerManagerTests::Task> taskToRemove
177 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout / 3));
178 timerManager.add(taskToRemove, taskToRemove->_timeout);
179 timerManager.add(taskToRemove, taskToRemove->_timeout * 2);
180
181 shared_ptr<TimerManagerTests::Task> task
182 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout));
183 timerManager.add(task, task->_timeout);
184
185 // Remove the first task (e.g. two timers) and wait until the other has completed
186 timerManager.remove(taskToRemove);
187 _monitor.wait(timeout * 2);
188
189 assert(!taskToRemove->_done);
190 assert(task->_done);
191
192 return true;
193 }
194
Francois Ferrand69603702017-09-11 12:09:40 +0200195 /**
196 * This test creates two tasks, removes the first one then waits for the second one. It then
197 * verifies that the timer manager properly clean up itself and the remaining orphaned timeout
198 * task when the manager goes out of scope and its destructor is called.
199 */
200 bool test03(int64_t timeout = 1000LL) {
201 TimerManager timerManager;
202 timerManager.threadFactory(shared_ptr<PlatformThreadFactory>(new PlatformThreadFactory()));
203 timerManager.start();
204 assert(timerManager.state() == TimerManager::STARTED);
205
206 Synchronized s(_monitor);
207
208 // Setup the two tasks
209 shared_ptr<TimerManagerTests::Task> taskToRemove
210 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout / 2));
211 TimerManager::Timer timer = timerManager.add(taskToRemove, taskToRemove->_timeout);
212
213 shared_ptr<TimerManagerTests::Task> task
214 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout));
215 timerManager.add(task, task->_timeout);
216
217 // Remove one task and wait until the other has completed
218 timerManager.remove(timer);
219 _monitor.wait(timeout * 2);
220
221 assert(!taskToRemove->_done);
222 assert(task->_done);
223
224 // Verify behavior when removing the removed task
225 try {
226 timerManager.remove(timer);
227 assert(0 == "ERROR: This remove should send a NoSuchTaskException exception.");
228 } catch (NoSuchTaskException&) {
229 }
230
231 return true;
232 }
233
234 /**
235 * This test creates one tasks, and tries to remove it after it has expired.
236 */
237 bool test04(int64_t timeout = 1000LL) {
238 TimerManager timerManager;
239 timerManager.threadFactory(shared_ptr<PlatformThreadFactory>(new PlatformThreadFactory()));
240 timerManager.start();
241 assert(timerManager.state() == TimerManager::STARTED);
242
243 Synchronized s(_monitor);
244
245 // Setup the task
246 shared_ptr<TimerManagerTests::Task> task
247 = shared_ptr<TimerManagerTests::Task>(new TimerManagerTests::Task(_monitor, timeout / 10));
248 TimerManager::Timer timer = timerManager.add(task, task->_timeout);
249
250 // Wait until the task has completed
251 _monitor.wait(timeout);
252
253 // Verify behavior when removing the expired task
254 try {
255 timerManager.remove(timer);
256 assert(0 == "ERROR: This remove should send a NoSuchTaskException exception.");
257 } catch (NoSuchTaskException&) {
258 }
259
260 return true;
261 }
262
Marc Slemko8a40a762006-07-19 17:46:50 +0000263 friend class TestTask;
264
265 Monitor _monitor;
266};
Marc Slemko8a40a762006-07-19 17:46:50 +0000267
Konrad Grochowski16a23a62014-11-13 15:33:38 +0100268}
269}
270}
271} // apache::thrift::concurrency