From f86623a70c70920dd8f06e292afa0d96067d057d Mon Sep 17 00:00:00 2001 From: Lisa Julia Nebel Date: Mon, 8 Jul 2024 16:56:19 +0200 Subject: [PATCH] Add test for failing tasklets --- tests/models/test_tasklets_failure.cpp | 135 +++++++++++++++++++++++++ 1 file changed, 135 insertions(+) create mode 100644 tests/models/test_tasklets_failure.cpp diff --git a/tests/models/test_tasklets_failure.cpp b/tests/models/test_tasklets_failure.cpp new file mode 100644 index 000000000..2a10bfdbf --- /dev/null +++ b/tests/models/test_tasklets_failure.cpp @@ -0,0 +1,135 @@ +// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- +// vi: set et ts=4 sw=4 sts=4: +/* + This file is part of the Open Porous Media project (OPM). + + OPM 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, either version 2 of the License, or + (at your option) any later version. + + OPM 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 OPM. If not, see . + + Consult the COPYING file in the top-level source directory of this + module for the precise wording of the license and the list of + copyright holders. +*/ +/*! + * \file + * + * \brief This file serves as an example of how to use the tasklet mechanism for + * asynchronous work, especially for tasklets that fail. + */ +#define BOOST_TEST_MODULE TASKLETS_FAILURE + +//#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "config.h" +#include + +std::mutex outputMutex; + +Opm::TaskletRunner *runner; + +class SleepTasklet : public Opm::TaskletInterface +{ +public: + SleepTasklet(int mseconds, int id) + : mseconds_(mseconds), + id_(id) + {} + + void run() override + { + assert(0 <= runner->workerThreadIndex() && runner->workerThreadIndex() < runner->numWorkerThreads()); + std::cout << "Sleep tasklet " << id_ << " of " << mseconds_ << " ms starting sleep on worker thread " << runner->workerThreadIndex() << std::endl; + std::this_thread::sleep_for(std::chrono::milliseconds(mseconds_)); + outputMutex.lock(); + std::cout << "Sleep tasklet " << id_ << " of " << mseconds_ << " ms completed by worker thread " << runner->workerThreadIndex() << std::endl; + outputMutex.unlock(); + } + +private: + int mseconds_; + int id_; +}; + +class FailingSleepTasklet : public Opm::TaskletInterface +{ +public: + FailingSleepTasklet(int mseconds) + : mseconds_(mseconds) + {} + void run() override + { + std::this_thread::sleep_for(std::chrono::milliseconds(mseconds_)); + outputMutex.lock(); + std::cout << "Failing sleep tasklet of " << mseconds_ << " ms failing now, on work thread " << runner->workerThreadIndex() << std::endl; + outputMutex.unlock(); + throw std::logic_error("Intentional failure for testing"); + } + +private: + int mseconds_; +}; + +void execute () { + int numWorkers = 2; + runner = new Opm::TaskletRunner(numWorkers); + + // the master thread is not a worker thread + assert(runner->workerThreadIndex() < 0); + assert(runner->numWorkerThreads() == numWorkers); + + // Dispatch some successful tasklets + for (int i = 0; i < 5; ++i) { + auto st = std::make_shared(10,i); + runner->dispatch(st); + } + + // Dispatch a failing tasklet + auto failingSleepTasklet = std::make_shared(100); + runner->dispatch(failingSleepTasklet); + + // Dispatch more successful tasklets + for (int i = 5; i < 10; ++i) { + auto st = std::make_shared(10,i); + runner->dispatch(st); + } + + std::cout << "before barrier" << std::endl; + runner->barrier(); + } +BOOST_AUTO_TEST_SUITE(Tasklets) +BOOST_AUTO_TEST_CASE(TASKLETS_FAILURE) { + pid_t pid = fork(); // Create a new process, such that this child process can call exit(EXIT_FAILURE) + if (pid == -1) { + BOOST_FAIL("Fork failed"); + } else if (pid == 0) { + // Child process + execute(); + _exit(0); // Should never reach here + } else { + // Parent process + std::cout << "Checking failure of child process with parent process, process id " << pid << std::endl; + int status; + waitpid(pid, &status, 0); + BOOST_CHECK(WIFEXITED(status)); // Check if the child process exited + BOOST_CHECK_EQUAL(WEXITSTATUS(status), EXIT_FAILURE); // Check if the exit status is EXIT_FAILURE + } +} +BOOST_AUTO_TEST_SUITE_END()