mirror of
https://github.com/pocoproject/poco.git
synced 2025-04-16 06:56:41 +02:00
375 lines
7.9 KiB
C++
375 lines
7.9 KiB
C++
//
|
|
// TimerTest.cpp
|
|
//
|
|
// Copyright (c) 2009, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "TimerTest.h"
|
|
#include "CppUnit/TestCaller.h"
|
|
#include "CppUnit/TestSuite.h"
|
|
#include "Poco/Util/Timer.h"
|
|
#include "Poco/Util/TimerTaskAdapter.h"
|
|
|
|
|
|
using Poco::Util::Timer;
|
|
using Poco::Util::TimerTask;
|
|
using Poco::Util::TimerTaskAdapter;
|
|
using Poco::Timestamp;
|
|
using Poco::Clock;
|
|
|
|
|
|
TimerTest::TimerTest(const std::string& name): CppUnit::TestCase(name)
|
|
{
|
|
}
|
|
|
|
|
|
TimerTest::~TimerTest()
|
|
{
|
|
}
|
|
|
|
|
|
void TimerTest::testScheduleTimestamp()
|
|
{
|
|
Timer timer;
|
|
|
|
Timestamp time;
|
|
time += 1000000;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
timer.schedule(pTask, time);
|
|
|
|
_event.wait();
|
|
assertTrue (pTask->lastExecution() >= time);
|
|
}
|
|
|
|
|
|
void TimerTest::testScheduleClock()
|
|
{
|
|
Timer timer;
|
|
|
|
// As reference
|
|
Timestamp time;
|
|
time += 1000000;
|
|
|
|
Clock clock;
|
|
clock += 1000000;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
timer.schedule(pTask, clock);
|
|
|
|
_event.wait();
|
|
assertTrue (pTask->lastExecution() >= time);
|
|
}
|
|
|
|
|
|
void TimerTest::testScheduleInterval()
|
|
{
|
|
Timer timer;
|
|
|
|
Timestamp time;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
timer.schedule(pTask, 500, 500);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 590000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1190000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1790000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
pTask->cancel();
|
|
assertTrue (pTask->isCancelled());
|
|
}
|
|
|
|
|
|
void TimerTest::testScheduleIntervalTimestamp()
|
|
{
|
|
Timer timer;
|
|
|
|
Timestamp time;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
Timestamp scheduleTime;
|
|
scheduleTime += 500 * 1000;
|
|
|
|
timer.schedule(pTask, scheduleTime, 500);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 590000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1190000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1790000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
pTask->cancel();
|
|
assertTrue (pTask->isCancelled());
|
|
}
|
|
|
|
|
|
void TimerTest::testScheduleIntervalClock()
|
|
{
|
|
Timer timer;
|
|
|
|
Timestamp time;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
Clock scheduleClock;
|
|
scheduleClock += 500 * 1000;
|
|
|
|
timer.schedule(pTask, scheduleClock, 500);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 590000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1190000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1790000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
pTask->cancel();
|
|
assertTrue (pTask->isCancelled());
|
|
}
|
|
|
|
|
|
void TimerTest::testScheduleAtFixedRate()
|
|
{
|
|
Timer timer;
|
|
|
|
Timestamp time;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
timer.scheduleAtFixedRate(pTask, 500, 500);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 500000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1000000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
_event.wait();
|
|
assertTrue (time.elapsed() >= 1500000);
|
|
assertTrue (pTask->lastExecution().elapsed() < 130000);
|
|
|
|
pTask->cancel();
|
|
assertTrue (pTask->isCancelled());
|
|
}
|
|
|
|
|
|
void TimerTest::testCancel()
|
|
{
|
|
Timer timer;
|
|
|
|
Timestamp time;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
assertTrue (pTask->lastExecution() == 0);
|
|
|
|
timer.scheduleAtFixedRate(pTask, 5000, 5000);
|
|
|
|
pTask->cancel();
|
|
assertTrue (pTask->isCancelled());
|
|
|
|
try
|
|
{
|
|
timer.scheduleAtFixedRate(pTask, 5000, 5000);
|
|
fail("must not reschedule a cancelled task");
|
|
}
|
|
catch (Poco::IllegalStateException&)
|
|
{
|
|
}
|
|
catch (Poco::Exception&)
|
|
{
|
|
fail("bad exception thrown");
|
|
}
|
|
}
|
|
|
|
|
|
void TimerTest::testCancelAllStop()
|
|
{
|
|
{
|
|
Timer timer;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
// We are scheduling a timer event in 100ms
|
|
Timestamp time;
|
|
time += 100000;
|
|
timer.schedule(pTask, time);
|
|
|
|
timer.cancel(false);
|
|
// Timer should fire in 100ms and onTimer has 100ms sleep in it.
|
|
// So we are waiting 2 times that plus a small buffer that to make sure that event was never executed.
|
|
bool timerExecuted = _event.tryWait(200 + 50);
|
|
assertFalse (timerExecuted);
|
|
}
|
|
|
|
assertTrue (true); // don't hang
|
|
}
|
|
|
|
|
|
void TimerTest::testCancelAllWaitStop()
|
|
{
|
|
{
|
|
Timer timer;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
// We are scheduling a timer event in 100ms
|
|
Timestamp time;
|
|
time += 100000;
|
|
timer.schedule(pTask, time);
|
|
|
|
timer.cancel(true);
|
|
// Timer should fire in 100ms and onTimer has 100ms sleep in it.
|
|
// So we are waiting 2 times that plus a small buffer that to make sure that event was never executed.
|
|
bool timerExecuted = _event.tryWait(200 + 50);
|
|
assertFalse (timerExecuted);
|
|
}
|
|
|
|
assertTrue (true); // don't hang
|
|
}
|
|
|
|
|
|
void TimerTest::testMultiCancelAllWaitStop()
|
|
{
|
|
Timer timer;
|
|
|
|
// We will schedule a task and wait for it to start.
|
|
// After that we will schedule 2 cancel Notifications, one async and the other sync.
|
|
// But we want to make sure that both are scheduled and present in internal queue, thus we need to wait for this
|
|
// first task to start.
|
|
Poco::Event startEvent;
|
|
Poco::Event canceledScheduledEvent;
|
|
timer.schedule(Timer::func([&startEvent, &canceledScheduledEvent]()
|
|
{
|
|
startEvent.set();
|
|
canceledScheduledEvent.wait();
|
|
Poco::Thread::sleep(100);
|
|
}), Poco::Clock());
|
|
// We wait for simple task to start.
|
|
startEvent.wait();
|
|
// Schedule async cancel notification.
|
|
timer.cancel();
|
|
// Now allow simple task to proceed to sleep, in other words give time for next cancel to block.
|
|
canceledScheduledEvent.set();
|
|
// Schedule sync cancel, now we should have 2 cancel notifications in internal queue.
|
|
timer.cancel(true);
|
|
|
|
assertTrue (true); // don't hang
|
|
}
|
|
|
|
|
|
void TimerTest::testFunc()
|
|
{
|
|
Timer timer;
|
|
|
|
std::atomic<int> count(0);
|
|
timer.schedule(Timer::func([&count]()
|
|
{
|
|
count++;
|
|
}), Poco::Clock());
|
|
Poco::Thread::sleep(100);
|
|
|
|
assertTrue (count == 1);
|
|
}
|
|
|
|
|
|
void TimerTest::testIdle()
|
|
{
|
|
Timer timer;
|
|
|
|
assertTrue (timer.idle());
|
|
|
|
Timestamp time;
|
|
time += 1000000;
|
|
|
|
TimerTask::Ptr pTask = new TimerTaskAdapter<TimerTest>(*this, &TimerTest::onTimer);
|
|
|
|
timer.schedule(pTask, time);
|
|
|
|
assertFalse (timer.idle());
|
|
|
|
_event.wait();
|
|
assertTrue (pTask->lastExecution() >= time);
|
|
|
|
assertTrue (timer.idle());
|
|
}
|
|
|
|
|
|
void TimerTest::setUp()
|
|
{
|
|
}
|
|
|
|
|
|
void TimerTest::tearDown()
|
|
{
|
|
}
|
|
|
|
|
|
void TimerTest::onTimer(TimerTask& task)
|
|
{
|
|
Poco::Thread::sleep(100);
|
|
_event.set();
|
|
}
|
|
|
|
|
|
CppUnit::Test* TimerTest::suite()
|
|
{
|
|
CppUnit::TestSuite* pSuite = new CppUnit::TestSuite("TimerTest");
|
|
|
|
CppUnit_addTest(pSuite, TimerTest, testScheduleTimestamp);
|
|
CppUnit_addTest(pSuite, TimerTest, testScheduleClock);
|
|
CppUnit_addTest(pSuite, TimerTest, testScheduleInterval);
|
|
CppUnit_addTest(pSuite, TimerTest, testScheduleIntervalTimestamp);
|
|
CppUnit_addTest(pSuite, TimerTest, testScheduleIntervalClock);
|
|
CppUnit_addTest(pSuite, TimerTest, testScheduleAtFixedRate);
|
|
CppUnit_addTest(pSuite, TimerTest, testCancel);
|
|
CppUnit_addTest(pSuite, TimerTest, testCancelAllStop);
|
|
CppUnit_addTest(pSuite, TimerTest, testCancelAllWaitStop);
|
|
CppUnit_addTest(pSuite, TimerTest, testMultiCancelAllWaitStop);
|
|
CppUnit_addTest(pSuite, TimerTest, testFunc);
|
|
CppUnit_addTest(pSuite, TimerTest, testIdle);
|
|
|
|
return pSuite;
|
|
}
|