diff options
| author | Mehmet Samet Duman <yongdohyun@projecttick.org> | 2026-04-02 18:51:45 +0300 |
|---|---|---|
| committer | Mehmet Samet Duman <yongdohyun@projecttick.org> | 2026-04-02 18:51:45 +0300 |
| commit | d3261e64152397db2dca4d691a990c6bc2a6f4dd (patch) | |
| tree | fac2f7be638651181a72453d714f0f96675c2b8b /archived/projt-launcher/tests/Task_test.cpp | |
| parent | 31b9a8949ed0a288143e23bf739f2eb64fdc63be (diff) | |
| download | Project-Tick-d3261e64152397db2dca4d691a990c6bc2a6f4dd.tar.gz Project-Tick-d3261e64152397db2dca4d691a990c6bc2a6f4dd.zip | |
NOISSUE add archived projects
Signed-off-by: Mehmet Samet Duman <yongdohyun@projecttick.org>
Diffstat (limited to 'archived/projt-launcher/tests/Task_test.cpp')
| -rw-r--r-- | archived/projt-launcher/tests/Task_test.cpp | 324 |
1 files changed, 324 insertions, 0 deletions
diff --git a/archived/projt-launcher/tests/Task_test.cpp b/archived/projt-launcher/tests/Task_test.cpp new file mode 100644 index 0000000000..8e3c2f8a33 --- /dev/null +++ b/archived/projt-launcher/tests/Task_test.cpp @@ -0,0 +1,324 @@ +// SPDX-License-Identifier: GPL-3.0-only +// SPDX-FileCopyrightText: 2026 Project Tick +// SPDX-FileContributor: Project Tick Team +/* + * ProjT Launcher - Minecraft Launcher + * Copyright (C) 2026 Project Tick + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, version 3. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ + +#include <QTest> +#include <QThread> +#include <QTimer> + +#include <tasks/ConcurrentTask.h> +#include <tasks/MultipleOptionsTask.h> +#include <tasks/SequentialTask.h> +#include <tasks/Task.h> + +#include <array> + +/* Does nothing. Only used for testing. */ +class BasicTask : public Task +{ + Q_OBJECT + + friend class TaskTest; + + public: + BasicTask(bool show_debug_log = true) : Task(show_debug_log) + {} + + private: + void executeTask() override + { + emitSucceeded(); + } +}; + +/* Does nothing. Only used for testing. */ +class BasicTask_MultiStep : public Task +{ + Q_OBJECT + + friend class TaskTest; + + private: + auto isMultiStep() const -> bool override + { + return true; + } + + void executeTask() override + {} +}; + +class BigConcurrentTask : public ConcurrentTask +{ + Q_OBJECT + + void executeNextSubTask() override + { + // This is here only to help fill the stack a bit more quickly (if there's an issue, of course :^)) + // Each tasks thus adds 1024 * 4 bytes to the stack, at the very least. + [[maybe_unused]] volatile std::array<uint32_t, 1024> some_data_on_the_stack{}; + + ConcurrentTask::executeNextSubTask(); + } +}; + +class BigConcurrentTaskThread : public QThread +{ + Q_OBJECT + + QTimer m_deadline; + void run() override + { + BigConcurrentTask big_task; + // This test is sensitive to runner performance and can be flaky on slower CI hardware. + m_deadline.setInterval(30000); + + // NOTE: Arbitrary value that manages to trigger a problem when there is one. + // Considering each tasks, in a problematic state, adds 1024 * 4 bytes to the stack, + // this number is enough to fill up 16 MiB of stack, more than enough to cause a problem. + static const unsigned s_num_tasks = 1 << 12; + for (unsigned i = 0; i < s_num_tasks; i++) + { + auto sub_task = makeShared<BasicTask>(false); + big_task.addTask(sub_task); + } + + connect(&big_task, &Task::finished, this, &QThread::quit); + connect(&m_deadline, + &QTimer::timeout, + this, + [this] + { + passed_the_deadline = true; + quit(); + }); + + if (thread() != QThread::currentThread()) + { + QMetaObject::invokeMethod(this, &BigConcurrentTaskThread::start_timer, Qt::QueuedConnection); + } + big_task.run(); + + exec(); + } + void start_timer() + { + m_deadline.start(); + } + + public: + bool passed_the_deadline = false; +}; + +class TaskTest : public QObject +{ + Q_OBJECT + + private slots: + void test_SetStatus_NoMultiStep() + { + BasicTask t; + QString status{ "test status" }; + + t.setStatus(status); + + QCOMPARE(t.getStatus(), status); + QCOMPARE(t.getStepProgress().isEmpty(), TaskStepProgressList{}.isEmpty()); + } + + void test_SetStatus_MultiStep() + { + BasicTask_MultiStep t; + QString status{ "test status" }; + + t.setStatus(status); + + QCOMPARE(t.getStatus(), status); + // Even though it is multi step, it does not override the getStepStatus method, + // so it should remain the same. + QCOMPARE(t.getStepProgress().isEmpty(), TaskStepProgressList{}.isEmpty()); + } + + void test_SetProgress() + { + BasicTask t; + int current = 42; + int total = 207; + + t.setProgress(current, total); + + QCOMPARE(t.getProgress(), current); + QCOMPARE(t.getTotalProgress(), total); + } + + void test_basicRun() + { + BasicTask t; + connect(&t, + &Task::finished, + [&t] + { QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); }); + t.start(); + + QVERIFY2(QTest::qWaitFor([&t]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); + } + + void test_basicConcurrentRun() + { + auto t1 = makeShared<BasicTask>(); + auto t2 = makeShared<BasicTask>(); + auto t3 = makeShared<BasicTask>(); + + ConcurrentTask t; + + t.addTask(t1); + t.addTask(t2); + t.addTask(t3); + + connect(&t, + &Task::finished, + [&t, &t1, &t2, &t3] + { + QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); + QVERIFY(t1->wasSuccessful()); + QVERIFY(t2->wasSuccessful()); + QVERIFY(t3->wasSuccessful()); + }); + + t.start(); + QVERIFY2(QTest::qWaitFor([&t]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); + } + + // Tests if starting new tasks after the 6 initial ones is working + void test_moreConcurrentRun() + { + auto t1 = makeShared<BasicTask>(); + auto t2 = makeShared<BasicTask>(); + auto t3 = makeShared<BasicTask>(); + auto t4 = makeShared<BasicTask>(); + auto t5 = makeShared<BasicTask>(); + auto t6 = makeShared<BasicTask>(); + auto t7 = makeShared<BasicTask>(); + auto t8 = makeShared<BasicTask>(); + auto t9 = makeShared<BasicTask>(); + + ConcurrentTask t; + + t.addTask(t1); + t.addTask(t2); + t.addTask(t3); + t.addTask(t4); + t.addTask(t5); + t.addTask(t6); + t.addTask(t7); + t.addTask(t8); + t.addTask(t9); + + connect(&t, + &Task::finished, + [&t, &t1, &t2, &t3, &t4, &t5, &t6, &t7, &t8, &t9] + { + QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); + QVERIFY(t1->wasSuccessful()); + QVERIFY(t2->wasSuccessful()); + QVERIFY(t3->wasSuccessful()); + QVERIFY(t4->wasSuccessful()); + QVERIFY(t5->wasSuccessful()); + QVERIFY(t6->wasSuccessful()); + QVERIFY(t7->wasSuccessful()); + QVERIFY(t8->wasSuccessful()); + QVERIFY(t9->wasSuccessful()); + }); + + t.start(); + QVERIFY2(QTest::qWaitFor([&t]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); + } + + void test_basicSequentialRun() + { + auto t1 = makeShared<BasicTask>(); + auto t2 = makeShared<BasicTask>(); + auto t3 = makeShared<BasicTask>(); + + SequentialTask t; + + t.addTask(t1); + t.addTask(t2); + t.addTask(t3); + + connect(&t, + &Task::finished, + [&t, &t1, &t2, &t3] + { + QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); + QVERIFY(t1->wasSuccessful()); + QVERIFY(t2->wasSuccessful()); + QVERIFY(t3->wasSuccessful()); + }); + + t.start(); + QVERIFY2(QTest::qWaitFor([&t]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); + } + + void test_basicMultipleOptionsRun() + { + auto t1 = makeShared<BasicTask>(); + auto t2 = makeShared<BasicTask>(); + auto t3 = makeShared<BasicTask>(); + + MultipleOptionsTask t; + + t.addTask(t1); + t.addTask(t2); + t.addTask(t3); + + connect(&t, + &Task::finished, + [&t, &t1, &t2, &t3] + { + QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); + QVERIFY(t1->wasSuccessful()); + QVERIFY(!t2->wasSuccessful()); + QVERIFY(!t3->wasSuccessful()); + }); + + t.start(); + QVERIFY2(QTest::qWaitFor([&t]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); + } + + void test_stackOverflowInConcurrentTask() + { + QEventLoop loop; + + BigConcurrentTaskThread thread; + + connect(&thread, &BigConcurrentTaskThread::finished, &loop, &QEventLoop::quit); + + thread.start(); + + loop.exec(); + + QVERIFY(!thread.passed_the_deadline); + } +}; + +QTEST_GUILESS_MAIN(TaskTest) + +#include "Task_test.moc" |
