From 48b7e69fd547baff1bc1ddc01e32a070dcee892c Mon Sep 17 00:00:00 2001 From: Gavin Halliday Date: Wed, 6 Nov 2024 10:14:31 +0000 Subject: [PATCH 1/2] HPCC-32954 Add unit tests for the jobqueue Signed-off-by: Gavin Halliday --- common/workunit/wujobq.cpp | 14 ++ common/workunit/wujobq.hpp | 1 + testing/unittests/CMakeLists.txt | 2 + testing/unittests/dalitests.cpp | 389 +++++++++++++++++++++++++++++++ 4 files changed, 406 insertions(+) diff --git a/common/workunit/wujobq.cpp b/common/workunit/wujobq.cpp index 47a35e8f1dd..cd1f0d5b330 100644 --- a/common/workunit/wujobq.cpp +++ b/common/workunit/wujobq.cpp @@ -1264,6 +1264,20 @@ class CJobQueue: public CJobQueueBase, implements IJobQueue } + IJobQueueItem *dequeue(int minPrio, unsigned timeout, unsigned prioritytransitiondelay) override + { + Owned item; + if (prioritytransitiondelay) + { + unsigned timeout = prioritytransitiondelay; + bool usePrevPrio = true; + item.setown(dodequeue(minPrio, timeout, usePrevPrio, nullptr)); + } + if (!item) + item.setown(dodequeue(minPrio, timeout-prioritytransitiondelay, false, nullptr)); + return item.getClear(); + } + void placeonqueue(sQueueData &qd, IJobQueueItem *qitem,unsigned idx) // takes ownership of qitem { Owned qi = qitem; diff --git a/common/workunit/wujobq.hpp b/common/workunit/wujobq.hpp index c3209904a50..4fdcda7a837 100644 --- a/common/workunit/wujobq.hpp +++ b/common/workunit/wujobq.hpp @@ -120,6 +120,7 @@ interface IJobQueue: extends IJobQueueConst virtual void connect(bool validateitemsessions)=0; // must be called before dequeueing // validateitemsessions ensures that all queue items have running session virtual IJobQueueItem *dequeue(unsigned timeout=INFINITE)=0; + virtual IJobQueueItem *dequeue(int minPrio, unsigned timeout, unsigned prioritytransitiondelay)=0; virtual void disconnect()=0; // signal no longer wil be dequeing (optional - done automatically on release) virtual void getStats(unsigned &connected,unsigned &waiting, unsigned &enqueued)=0; // this not quick as validates clients still running virtual bool waitStatsChange(unsigned timeout)=0; diff --git a/testing/unittests/CMakeLists.txt b/testing/unittests/CMakeLists.txt index 59498fb0d2f..ee8278a30c3 100644 --- a/testing/unittests/CMakeLists.txt +++ b/testing/unittests/CMakeLists.txt @@ -78,6 +78,7 @@ include_directories ( ./../../dali/base ./../../system/security/shared ./../../common/deftype + ./../../common/workunit ./../../system/security/cryptohelper ./../../configuration/configmgr/configmgrlib ${HPCC_SOURCE_DIR}/system/masking/include @@ -118,6 +119,7 @@ target_link_libraries ( unittests esphttp esdllib logginglib + workunit ${CppUnit_LIBRARIES} ) diff --git a/testing/unittests/dalitests.cpp b/testing/unittests/dalitests.cpp index 65d76ea383e..1f7b46c9742 100644 --- a/testing/unittests/dalitests.cpp +++ b/testing/unittests/dalitests.cpp @@ -31,11 +31,14 @@ #include "dasds.hpp" #include "danqs.hpp" #include "dautils.hpp" +#include "wujobq.hpp" #include #include #include +#include "jthread.hpp" + #include "unittests.hpp" #include "sysinfologger.hpp" @@ -3262,4 +3265,390 @@ class DaliSysInfoLoggerTester : public CppUnit::TestFixture CPPUNIT_TEST_SUITE_REGISTRATION( DaliSysInfoLoggerTester ); CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( DaliSysInfoLoggerTester, "DaliSysInfoLoggerTester" ); + + + +static constexpr bool traceJobQueue = false; +static unsigned jobQueueStartTick; +//The following allows the tests to be slowed down to make it easier to debug problems +static constexpr unsigned tickScaling = 1; +static unsigned getJobQueueTick() +{ + return (msTick() - jobQueueStartTick) / tickScaling; +} +static void JobQueueSleep(unsigned ms) +{ + MilliSleep(ms * tickScaling); +} +class JobQueueTester : public CppUnit::TestFixture +{ + CPPUNIT_TEST_SUITE(JobQueueTester); + CPPUNIT_TEST(testInit); + CPPUNIT_TEST(testSingle); + CPPUNIT_TEST(testDouble); + CPPUNIT_TEST(testMany); + CPPUNIT_TEST(testCleanup); + CPPUNIT_TEST_SUITE_END(); + + struct JobEntry + { + unsigned delayMs; + const char * name; + unsigned processingMs; + int priority; + }; + + class JobProcessor : public Thread + { + public: + JobProcessor(Semaphore & _startedSem, Semaphore & _processedSem, IJobQueue * _queue, unsigned _id) + : startedSem(_startedSem), processedSem(_processedSem), queue(_queue), id(_id) + { + } + + virtual int run() override + { + startedSem.signal(); + try + { + processAll(); + } + catch (IException * _e) + { + e.setown(_e); + } + return 0; + } + + bool processItem(IJobQueueItem * item) + { + assertex(item); + const char * name = item->queryWUID(); + if (traceJobQueue) + DBGLOG("===%s===@%u", name, getJobQueueTick()); + if (name[0] == '!') + return false; + output.append(name); + if (!log.isEmpty()) + log.append(","); + log.append(name).append("@").append(getJobQueueTick()); + unsigned delay = item->getPort(); + JobQueueSleep(delay); + processedSem.signal(); + return true; + } + + const char * queryOutput() + { + if (e) + throw e.getClear(); + return output.str(); + } + + const char * queryLog() + { + return log.str(); + } + + virtual void processAll() = 0; + + protected: + Semaphore & startedSem; + Semaphore & processedSem; + Linked queue; + StringBuffer output; + StringBuffer log; + Owned e; + unsigned id; + }; + + // Read the first (highest priority) item on the queue - no minimum priority + class StandardJobProcessor : public JobProcessor + { + public: + StandardJobProcessor(Semaphore & _startedSem, Semaphore & _processedSem, IJobQueue * _queue, unsigned _id) + : JobProcessor(_startedSem, _processedSem, _queue, _id) + { + } + + virtual void processAll() override + { + for (;;) + { + Owned item = queue->dequeue(); + if (!processItem(item)) + break; + } + } + + }; + + // Read the first item on the queue, but wait for 200ms to see if there is an item queued with a prioprity >= the last item that was dequeued. + // This is only used for bare metal, and I'm not sure the semantics are very helpful. I think NewThor may be better approach. + class ThorJobProcessor : public JobProcessor + { + public: + ThorJobProcessor(Semaphore & _startedSem, Semaphore & _processedSem, IJobQueue * _queue, unsigned _id) + : JobProcessor(_startedSem, _processedSem, _queue, _id) + { + } + + virtual void processAll() override + { + for (;;) + { + Owned item = queue->dequeue(0, INFINITE, 200*tickScaling); + bool ret = processItem(item); + if (!ret) + break; + } + } + }; + + // For 200ms check to see if there is an item queued with the same priority that this thread last dequeued. Then wait for any item. + // This means if there is a single high priority job, the other threads do not wait for that high priority job. + class NewThorJobProcessor : public JobProcessor + { + public: + NewThorJobProcessor(Semaphore & _startedSem, Semaphore & _processedSem, IJobQueue * _queue, unsigned _id) + : JobProcessor(_startedSem, _processedSem, _queue, _id) + { + } + + virtual void processAll() override + { + for (;;) + { + Owned item = queue->dequeue(lastPrio, 200*tickScaling, 0); + if (!item) + item.setown(queue->dequeue(0, INFINITE, 0)); + lastPrio = item->getPriority(); + bool ret = processItem(item); + if (!ret) + break; + } + } + + protected: + int lastPrio = 0; + + }; + + enum JobProcessorType + { + StandardProcessor, + ThorProcessor, + NewThorProcessor, + }; + + void testInit() + { + daliClientInit(); + } + + void testCleanup() + { + daliClientEnd(); + } + + void runTestCase(const char * name, const std::initializer_list & jobs, const std::initializer_list & processors, const std::initializer_list & expectedResults, bool uniqueQueues) + { + Owned queue = createJobQueue("JobQueueTester"); + Semaphore startedSem; + Semaphore processedSem; + + CIArrayOf jobProcessors; + for (auto & processor : processors) + { + JobProcessor * cur = nullptr; + Owned localQueue; + IJobQueue * processorQueue = queue; + if (uniqueQueues) + { + localQueue.setown(createJobQueue("JobQueueTester")); + processorQueue = localQueue; + } + + switch (processor) + { + case StandardProcessor: + cur = new StandardJobProcessor(startedSem, processedSem, processorQueue, jobProcessors.ordinality()); + break; + case ThorProcessor: + cur = new ThorJobProcessor(startedSem, processedSem, processorQueue, jobProcessors.ordinality()); + break; + case NewThorProcessor: + cur = new NewThorJobProcessor(startedSem, processedSem, processorQueue, jobProcessors.ordinality()); + break; + default: + UNIMPLEMENTED; + } + jobProcessors.append(*cur); + cur->start(true); + } + + for (auto & processor : processors) + startedSem.wait(); + + IArrayOf conversations; + jobQueueStartTick = msTick(); + for (auto & job : jobs) + { + JobQueueSleep(job.delayMs); + if (traceJobQueue) + DBGLOG("Add (%s, %d, %d) @%u", job.name, job.delayMs, job.processingMs, getJobQueueTick()); + Owned item = createJobQueueItem(job.name); + item->setPort(job.processingMs); + item->setPriority(job.priority); + + queue->enqueue(item.getClear()); + } + + ForEachItemIn(i1, jobProcessors) + { + if (traceJobQueue) + DBGLOG("Add (eoj) @%u", getJobQueueTick()); + + //The queue code dedups by "wuid", so we need to add a unique "stop" entry + std::string end = std::string("!") + std::to_string(i1); + Owned item = createJobQueueItem(end.c_str()); + queue->enqueue(item.getClear()); + } + + ForEachItemIn(i2, jobProcessors) + { + if (traceJobQueue) + DBGLOG("Wait for %u", i2); + jobProcessors.item(i2).join(); + } + + DBGLOG("%s:%s, %ums", name, uniqueQueues ? " unique queues" : "", getJobQueueTick()); + ForEachItemIn(i3, jobProcessors) + { + JobProcessor & cur = jobProcessors.item(i3); + DBGLOG(" Result: '%s' '%s'", cur.queryOutput(), cur.queryLog()); +// if (i3 < expectedResults.size()) +// CPPUNIT_ASSERT_EQUAL(std::string(expectedResults.begin()[i3]), std::string(cur.queryOutput())); + } + } + + void runTestCaseX2(const char * name, const std::initializer_list & jobs, const std::initializer_list & processors, const std::initializer_list & expectedResults) + { + runTestCase(name, jobs, processors, expectedResults, false); + runTestCase(name, jobs, processors, expectedResults, true); + } + + static constexpr std::initializer_list singleWuTest = { + { 0, "a", 90, 0 }, + { 100, "b", 90, 0 }, + { 100, "c", 90, 0 }, + { 100, "d", 90, 0 }, + }; + + static constexpr std::initializer_list twoWuTest = { + { 0, "a", 90, 0 }, + { 50, "A", 90, 0}, + { 50, "b", 90, 0 }, + { 50, "B", 90, 0 }, + { 50, "c", 90, 0 }, + { 50, "C", 90, 0 }, + { 50, "d", 90, 0 }, + { 50, "D", 90, 0 }, + }; + + static constexpr std::initializer_list lowHighTest = { + { 0, "a", 90, 0 }, + { 50, "A", 90, 1}, + { 50, "b", 90, 0 }, + { 50, "B", 90, 1 }, + { 50, "c", 90, 0 }, + { 50, "C", 90, 1 }, + { 50, "d", 90, 0 }, + { 50, "D", 90, 1 }, + }; + + static constexpr std::initializer_list lowHigh2Test = { + { 0, "a", 90, 0 }, + { 50, "A", 90, 1}, + { 10, "b", 90, 0 }, + { 10, "B", 90, 1 }, + { 10, "c", 90, 0 }, + { 10, "C", 90, 1 }, + { 10, "d", 90, 0 }, + { 10, "D", 90, 1 }, + }; + + static constexpr std::initializer_list lowHigh3Test = { + { 0, "a", 90, 0 }, + { 50, "A", 90, 1}, + { 10, "b", 90, 0 }, + }; + + static constexpr std::initializer_list dripFeedTest = { + { 0, "a", 10, 0 }, + { 100, "b", 10, 0}, + { 100, "c", 10, 0}, + { 100, "d", 10, 0}, + { 100, "e", 10, 0}, + { 100, "f", 10, 0}, + { 100, "g", 10, 0}, + { 100, "h", 10, 0}, + { 100, "i", 10, 0}, + { 100, "j", 10, 0}, + }; + + static constexpr std::initializer_list drip2FeedTest = { + { 0, "a", 60, 0 }, + { 50, "b", 60, 0}, + { 50, "c", 60, 0}, + { 50, "d", 60, 0}, + { 50, "e", 60, 0}, + { 50, "f", 60, 0}, + { 50, "g", 60, 0}, + { 50, "h", 60, 0}, + { 50, "i", 60, 0}, + { 50, "j", 60, 0}, + { 50, "k", 60, 0}, + { 50, "l", 60, 0}, + { 50, "m", 60, 0}, + { 50, "n", 60, 0}, + { 50, "o", 60, 0}, + }; + + void testSingle() + { + runTestCase("1 wu, 1 standard", singleWuTest, { StandardProcessor }, { "abcd" }, false); + runTestCase("2 wu, 1 standard", twoWuTest, { StandardProcessor }, { "aAbBcCdD" }, false); + runTestCase("lo hi wu, 1 standard", lowHighTest, { StandardProcessor }, { "aABCDbcd" }, false); + runTestCase("lo hi2 wu, 1 standard", lowHigh2Test, { StandardProcessor }, { "aABCDbcd" }, false); + runTestCase("lo hi2 wu, 1 thor", lowHigh2Test, { ThorProcessor }, {}, false); + runTestCase("lo hi2 wu, 1 newthor", lowHigh2Test, { NewThorProcessor }, {}, false); + runTestCase("drip wu, 1 std", dripFeedTest, { StandardProcessor }, {}, false); + + } + + void testDouble() + { + runTestCaseX2("2 wu, 2 standard", twoWuTest, { StandardProcessor, StandardProcessor }, { "abcd", "ABCD" }); + runTestCaseX2("lo hi wu, 2 standard", lowHighTest, { StandardProcessor, StandardProcessor }, { "aBDc" "ACbd" }); + runTestCaseX2("lo hi2 wu, 2 standard", lowHigh2Test, { StandardProcessor, StandardProcessor }, { "a"}); + runTestCaseX2("lo hi2 wu, 2 thor", lowHigh2Test, { ThorProcessor, ThorProcessor }, {}); + runTestCaseX2("lo hi2 wu, 2 newthor", lowHigh2Test, { NewThorProcessor, NewThorProcessor }, {}); + + runTestCaseX2("lo hi3 wu, 2 thor", lowHigh3Test, { ThorProcessor, ThorProcessor }, {}); + runTestCaseX2("lo hi3 wu, 2 newthor", lowHigh3Test, { NewThorProcessor, NewThorProcessor }, {}); + runTestCaseX2("drip wu, 2 std", dripFeedTest, { StandardProcessor, StandardProcessor }, {}); + runTestCaseX2("drip wu, 2 newthor", dripFeedTest, { NewThorProcessor, NewThorProcessor }, {}); + } + + void testMany() + { + runTestCaseX2("drip wu, 3 std", dripFeedTest, { StandardProcessor, StandardProcessor, StandardProcessor }, {}); + runTestCaseX2("drip2 wu, 3 std", drip2FeedTest, { StandardProcessor, StandardProcessor, StandardProcessor }, {}); + } +}; + +CPPUNIT_TEST_SUITE_REGISTRATION( JobQueueTester ); +CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( JobQueueTester, "JobQueueTester" ); + #endif // _USE_CPPUNIT From cd3fbd6ecb36a5c72155e71c7781f6791501bd4e Mon Sep 17 00:00:00 2001 From: Gavin Halliday Date: Tue, 7 Jan 2025 10:15:31 +0000 Subject: [PATCH 2/2] Changes following review Signed-off-by: Gavin Halliday --- common/workunit/wujobq.cpp | 3 +- testing/unittests/dalitests.cpp | 54 ++++++++++++++++++++++----------- 2 files changed, 38 insertions(+), 19 deletions(-) diff --git a/common/workunit/wujobq.cpp b/common/workunit/wujobq.cpp index cd1f0d5b330..ecd34fbdee5 100644 --- a/common/workunit/wujobq.cpp +++ b/common/workunit/wujobq.cpp @@ -1269,9 +1269,8 @@ class CJobQueue: public CJobQueueBase, implements IJobQueue Owned item; if (prioritytransitiondelay) { - unsigned timeout = prioritytransitiondelay; bool usePrevPrio = true; - item.setown(dodequeue(minPrio, timeout, usePrevPrio, nullptr)); + item.setown(dodequeue(minPrio, prioritytransitiondelay, usePrevPrio, nullptr)); } if (!item) item.setown(dodequeue(minPrio, timeout-prioritytransitiondelay, false, nullptr)); diff --git a/testing/unittests/dalitests.cpp b/testing/unittests/dalitests.cpp index 1f7b46c9742..444f2117e8c 100644 --- a/testing/unittests/dalitests.cpp +++ b/testing/unittests/dalitests.cpp @@ -3269,20 +3269,20 @@ CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( DaliSysInfoLoggerTester, "DaliSysInfoLogg static constexpr bool traceJobQueue = false; -static unsigned jobQueueStartTick; +static unsigned jobQueueStartTick = 0; //The following allows the tests to be slowed down to make it easier to debug problems static constexpr unsigned tickScaling = 1; static unsigned getJobQueueTick() { return (msTick() - jobQueueStartTick) / tickScaling; } -static void JobQueueSleep(unsigned ms) +static void jobQueueSleep(unsigned ms) { MilliSleep(ms * tickScaling); } -class JobQueueTester : public CppUnit::TestFixture +class DaliJobQueueTester : public CppUnit::TestFixture { - CPPUNIT_TEST_SUITE(JobQueueTester); + CPPUNIT_TEST_SUITE(DaliJobQueueTester); CPPUNIT_TEST(testInit); CPPUNIT_TEST(testSingle); CPPUNIT_TEST(testDouble); @@ -3303,8 +3303,8 @@ class JobQueueTester : public CppUnit::TestFixture public: JobProcessor(Semaphore & _startedSem, Semaphore & _processedSem, IJobQueue * _queue, unsigned _id) : startedSem(_startedSem), processedSem(_processedSem), queue(_queue), id(_id) - { - } + { + } virtual int run() override { @@ -3333,7 +3333,7 @@ class JobQueueTester : public CppUnit::TestFixture log.append(","); log.append(name).append("@").append(getJobQueueTick()); unsigned delay = item->getPort(); - JobQueueSleep(delay); + jobQueueSleep(delay); processedSem.signal(); return true; } @@ -3453,7 +3453,7 @@ class JobQueueTester : public CppUnit::TestFixture void runTestCase(const char * name, const std::initializer_list & jobs, const std::initializer_list & processors, const std::initializer_list & expectedResults, bool uniqueQueues) { - Owned queue = createJobQueue("JobQueueTester"); + Owned queue = createJobQueue("DaliJobQueueTester"); Semaphore startedSem; Semaphore processedSem; @@ -3465,7 +3465,7 @@ class JobQueueTester : public CppUnit::TestFixture IJobQueue * processorQueue = queue; if (uniqueQueues) { - localQueue.setown(createJobQueue("JobQueueTester")); + localQueue.setown(createJobQueue("DaliJobQueueTester")); processorQueue = localQueue; } @@ -3494,7 +3494,7 @@ class JobQueueTester : public CppUnit::TestFixture jobQueueStartTick = msTick(); for (auto & job : jobs) { - JobQueueSleep(job.delayMs); + jobQueueSleep(job.delayMs); if (traceJobQueue) DBGLOG("Add (%s, %d, %d) @%u", job.name, job.delayMs, job.processingMs, getJobQueueTick()); Owned item = createJobQueueItem(job.name); @@ -3527,8 +3527,28 @@ class JobQueueTester : public CppUnit::TestFixture { JobProcessor & cur = jobProcessors.item(i3); DBGLOG(" Result: '%s' '%s'", cur.queryOutput(), cur.queryLog()); -// if (i3 < expectedResults.size()) -// CPPUNIT_ASSERT_EQUAL(std::string(expectedResults.begin()[i3]), std::string(cur.queryOutput())); + + //If expected results are provided, check that the result matches one of them (it is undefined which + //processor will match which result) + if (expectedResults.size()) + { + bool matched = false; + StringBuffer expectedText; + for (auto & expected : expectedResults) + { + if (streq(expected, cur.queryOutput())) + { + matched = true; + break; + } + expectedText.append(", ").append(expected); + } + if (!matched) + { + DBGLOG("Result does not match any expected: %s", expectedText.str()+2); + CPPUNIT_ASSERT_MESSAGE("Result does not match any of the expected results", false); + } + } } } @@ -3630,9 +3650,9 @@ class JobQueueTester : public CppUnit::TestFixture void testDouble() { runTestCaseX2("2 wu, 2 standard", twoWuTest, { StandardProcessor, StandardProcessor }, { "abcd", "ABCD" }); - runTestCaseX2("lo hi wu, 2 standard", lowHighTest, { StandardProcessor, StandardProcessor }, { "aBDc" "ACbd" }); - runTestCaseX2("lo hi2 wu, 2 standard", lowHigh2Test, { StandardProcessor, StandardProcessor }, { "a"}); - runTestCaseX2("lo hi2 wu, 2 thor", lowHigh2Test, { ThorProcessor, ThorProcessor }, {}); + runTestCaseX2("lo hi wu, 2 standard", lowHighTest, { StandardProcessor, StandardProcessor }, { "abcd", "ABCD" }); + runTestCaseX2("lo hi2 wu, 2 standard", lowHigh2Test, { StandardProcessor, StandardProcessor }, { "aBDc", "ACbd" }); + runTestCaseX2("lo hi2 wu, 2 thor", lowHigh2Test, { ThorProcessor, ThorProcessor }, { "aBDc", "ACbd" }); runTestCaseX2("lo hi2 wu, 2 newthor", lowHigh2Test, { NewThorProcessor, NewThorProcessor }, {}); runTestCaseX2("lo hi3 wu, 2 thor", lowHigh3Test, { ThorProcessor, ThorProcessor }, {}); @@ -3648,7 +3668,7 @@ class JobQueueTester : public CppUnit::TestFixture } }; -CPPUNIT_TEST_SUITE_REGISTRATION( JobQueueTester ); -CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( JobQueueTester, "JobQueueTester" ); +CPPUNIT_TEST_SUITE_REGISTRATION( DaliJobQueueTester ); +CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( DaliJobQueueTester, "DaliJobQueueTester" ); #endif // _USE_CPPUNIT