// // NotificationQueueTest.cpp // // Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH. // and Contributors. // // SPDX-License-Identifier: BSL-1.0 // #include "NotificationQueueTest.h" #include "CppUnit/TestCaller.h" #include "CppUnit/TestSuite.h" #include "Poco/NotificationQueue.h" #include "Poco/Notification.h" #include "Poco/Thread.h" #include "Poco/Runnable.h" #include "Poco/RunnableAdapter.h" #include "Poco/Random.h" using Poco::NotificationQueue; using Poco::Notification; using Poco::Thread; using Poco::RunnableAdapter; namespace { class QTestNotification: public Notification { public: QTestNotification(const std::string& data): _data(data) { } ~QTestNotification() { } const std::string& data() const { return _data; } private: std::string _data; }; } NotificationQueueTest::NotificationQueueTest(const std::string& name): CppUnit::TestCase(name) { } NotificationQueueTest::~NotificationQueueTest() { } void NotificationQueueTest::testQueueDequeue() { NotificationQueue queue; assertTrue (queue.empty()); assertTrue (queue.size() == 0); Notification* pNf = queue.dequeueNotification(); assertNullPtr(pNf); queue.enqueueNotification(new Notification); assertTrue (!queue.empty()); assertTrue (queue.size() == 1); pNf = queue.dequeueNotification(); assertNotNullPtr(pNf); assertTrue (queue.empty()); assertTrue (queue.size() == 0); pNf->release(); queue.enqueueNotification(new QTestNotification("first")); queue.enqueueNotification(new QTestNotification("second")); assertTrue (!queue.empty()); assertTrue (queue.size() == 2); QTestNotification* pTNf = dynamic_cast(queue.dequeueNotification()); assertNotNullPtr(pTNf); assertTrue (pTNf->data() == "first"); pTNf->release(); assertTrue (!queue.empty()); assertTrue (queue.size() == 1); pTNf = dynamic_cast(queue.dequeueNotification()); assertNotNullPtr(pTNf); assertTrue (pTNf->data() == "second"); pTNf->release(); assertTrue (queue.empty()); assertTrue (queue.size() == 0); pNf = queue.dequeueNotification(); assertNullPtr(pNf); } void NotificationQueueTest::testQueueDequeueUrgent() { NotificationQueue queue; queue.enqueueNotification(new QTestNotification("first")); queue.enqueueNotification(new QTestNotification("second")); queue.enqueueUrgentNotification(new QTestNotification("third")); assertTrue (!queue.empty()); assertTrue (queue.size() == 3); QTestNotification* pTNf = dynamic_cast(queue.dequeueNotification()); assertNotNullPtr(pTNf); assertTrue (pTNf->data() == "third"); pTNf->release(); assertTrue (!queue.empty()); assertTrue (queue.size() == 2); pTNf = dynamic_cast(queue.dequeueNotification()); assertTrue (pTNf->data() == "first"); pTNf->release(); assertTrue (!queue.empty()); assertTrue (queue.size() == 1); pTNf = dynamic_cast(queue.dequeueNotification()); assertNotNullPtr(pTNf); assertTrue (pTNf->data() == "second"); pTNf->release(); assertTrue (queue.empty()); assertTrue (queue.size() == 0); Notification* pNf = queue.dequeueNotification(); assertNullPtr(pNf); } void NotificationQueueTest::testWaitDequeue() { NotificationQueue queue; queue.enqueueNotification(new QTestNotification("third")); queue.enqueueNotification(new QTestNotification("fourth")); assertTrue (!queue.empty()); assertTrue (queue.size() == 2); QTestNotification* pTNf = dynamic_cast(queue.waitDequeueNotification(10)); assertNotNullPtr(pTNf); assertTrue (pTNf->data() == "third"); pTNf->release(); assertTrue (!queue.empty()); assertTrue (queue.size() == 1); pTNf = dynamic_cast(queue.waitDequeueNotification(10)); assertNotNullPtr(pTNf); assertTrue (pTNf->data() == "fourth"); pTNf->release(); assertTrue (queue.empty()); assertTrue (queue.size() == 0); Notification* pNf = queue.waitDequeueNotification(10); assertNullPtr(pNf); } void NotificationQueueTest::testThreads() { const int NOTIFICATION_COUNT = 5000; Thread t1("thread1"); Thread t2("thread2"); Thread t3("thread3"); RunnableAdapter ra(*this, &NotificationQueueTest::work); t1.start(ra); t2.start(ra); t3.start(ra); for (int i = 0; i < NOTIFICATION_COUNT; ++i) { _queue.enqueueNotification(new Notification); } while (!_queue.empty()) Thread::sleep(50); Thread::sleep(20); _queue.wakeUpAll(); t1.join(); t2.join(); t3.join(); assertTrue (_handled.size() == NOTIFICATION_COUNT); assertTrue (_handled.count("thread1") > 0); assertTrue (_handled.count("thread2") > 0); assertTrue (_handled.count("thread3") > 0); } void NotificationQueueTest::testDefaultQueue() { NotificationQueue& queue = NotificationQueue::defaultQueue(); assertTrue (queue.empty()); assertTrue (queue.size() == 0); } void NotificationQueueTest::setUp() { _handled.clear(); } void NotificationQueueTest::tearDown() { } void NotificationQueueTest::work() { Poco::Random rnd; Thread::sleep(50); Notification* pNf = _queue.waitDequeueNotification(); while (pNf) { pNf->release(); _mutex.lock(); _handled.insert(Thread::current()->name()); _mutex.unlock(); Thread::sleep(rnd.next(5)); pNf = _queue.waitDequeueNotification(); } } CppUnit::Test* NotificationQueueTest::suite() { CppUnit::TestSuite* pSuite = new CppUnit::TestSuite("NotificationQueueTest"); CppUnit_addTest(pSuite, NotificationQueueTest, testQueueDequeue); CppUnit_addTest(pSuite, NotificationQueueTest, testQueueDequeueUrgent); CppUnit_addTest(pSuite, NotificationQueueTest, testWaitDequeue); CppUnit_addTest(pSuite, NotificationQueueTest, testThreads); CppUnit_addTest(pSuite, NotificationQueueTest, testDefaultQueue); return pSuite; }