2024-10-28 15:57:39 +01:00
|
|
|
|
// Luanti
|
|
|
|
|
// SPDX-License-Identifier: LGPL-2.1-or-later
|
|
|
|
|
// Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
2017-08-17 22:19:39 +02:00
|
|
|
|
#pragma once
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
2023-12-15 12:23:32 +03:00
|
|
|
|
#include <functional>
|
2015-04-26 01:24:19 -04:00
|
|
|
|
#include <exception>
|
2023-12-15 12:23:32 +03:00
|
|
|
|
#include <sstream>
|
2015-04-26 01:24:19 -04:00
|
|
|
|
#include <vector>
|
|
|
|
|
|
2025-01-04 12:39:34 +01:00
|
|
|
|
#include "irrlichttypes_bloated.h"
|
2015-04-26 01:24:19 -04:00
|
|
|
|
#include "porting.h"
|
|
|
|
|
#include "filesys.h"
|
|
|
|
|
#include "mapnode.h"
|
|
|
|
|
|
2023-12-15 12:23:32 +03:00
|
|
|
|
class TestFailedException { // don’t derive from std::exception to avoid accidental catch
|
|
|
|
|
public:
|
|
|
|
|
TestFailedException(std::string in_message, const char *in_file, int in_line)
|
|
|
|
|
: message(std::move(in_message))
|
|
|
|
|
, file(fs::GetFilenameFromPath(in_file))
|
|
|
|
|
, line(in_line)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
const std::string message;
|
|
|
|
|
const std::string file;
|
|
|
|
|
const int line;
|
2015-04-26 01:24:19 -04:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Runs a unit test and reports results
|
2023-12-15 12:23:32 +03:00
|
|
|
|
#define TEST(fxn, ...) runTest(#fxn, [&] () { fxn(__VA_ARGS__); });
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
|
|
|
|
// Asserts the specified condition is true, or fails the current unit test
|
2023-12-15 12:23:32 +03:00
|
|
|
|
#define UASSERT(x) \
|
|
|
|
|
if (!(x)) { \
|
|
|
|
|
throw TestFailedException(#x, __FILE__, __LINE__); \
|
2017-08-19 22:23:47 +02:00
|
|
|
|
}
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
|
|
|
|
// Asserts the specified condition is true, or fails the current unit test
|
|
|
|
|
// and prints the format specifier fmt
|
2023-12-15 12:23:32 +03:00
|
|
|
|
#define UTEST(x, fmt, ...) \
|
|
|
|
|
if (!(x)) { \
|
|
|
|
|
char utest_buf[1024]; \
|
|
|
|
|
snprintf(utest_buf, sizeof(utest_buf), fmt, __VA_ARGS__); \
|
|
|
|
|
throw TestFailedException(utest_buf, __FILE__, __LINE__); \
|
2017-08-19 22:23:47 +02:00
|
|
|
|
}
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
|
|
|
|
// Asserts the comparison specified by CMP is true, or fails the current unit test
|
2023-12-15 12:23:32 +03:00
|
|
|
|
#define UASSERTCMP(T, CMP, actual, expected) { \
|
|
|
|
|
T a = (actual); \
|
|
|
|
|
T e = (expected); \
|
|
|
|
|
if (!(a CMP e)) { \
|
|
|
|
|
std::ostringstream message; \
|
|
|
|
|
message << #actual " " #CMP " " #expected; \
|
|
|
|
|
message << std::endl << " actual : " << a; \
|
|
|
|
|
message << std::endl << " expected: " << e; \
|
|
|
|
|
throw TestFailedException(message.str(), __FILE__, __LINE__); \
|
|
|
|
|
} \
|
2017-08-20 13:30:50 +02:00
|
|
|
|
}
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
|
|
|
|
#define UASSERTEQ(T, actual, expected) UASSERTCMP(T, ==, actual, expected)
|
|
|
|
|
|
|
|
|
|
// UASSERTs that the specified exception occurs
|
2017-08-20 13:30:50 +02:00
|
|
|
|
#define EXCEPTION_CHECK(EType, code) { \
|
2015-04-26 01:24:19 -04:00
|
|
|
|
bool exception_thrown = false; \
|
|
|
|
|
try { \
|
|
|
|
|
code; \
|
|
|
|
|
} catch (EType &e) { \
|
|
|
|
|
exception_thrown = true; \
|
|
|
|
|
} \
|
|
|
|
|
UASSERT(exception_thrown); \
|
2017-08-20 13:30:50 +02:00
|
|
|
|
}
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
|
|
|
|
class IGameDef;
|
|
|
|
|
|
|
|
|
|
class TestBase {
|
|
|
|
|
public:
|
|
|
|
|
bool testModule(IGameDef *gamedef);
|
2015-05-08 00:05:08 -04:00
|
|
|
|
std::string getTestTempDirectory();
|
|
|
|
|
std::string getTestTempFile();
|
|
|
|
|
|
2015-04-26 01:24:19 -04:00
|
|
|
|
virtual void runTests(IGameDef *gamedef) = 0;
|
|
|
|
|
virtual const char *getName() = 0;
|
|
|
|
|
|
|
|
|
|
u32 num_tests_failed;
|
|
|
|
|
u32 num_tests_run;
|
2015-05-08 00:05:08 -04:00
|
|
|
|
|
2023-12-15 12:23:32 +03:00
|
|
|
|
void runTest(const char *name, std::function<void()> &&test);
|
|
|
|
|
|
2015-05-08 00:05:08 -04:00
|
|
|
|
private:
|
|
|
|
|
std::string m_test_dir;
|
2015-04-26 01:24:19 -04:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class TestManager {
|
|
|
|
|
public:
|
|
|
|
|
static std::vector<TestBase *> &getTestModules()
|
|
|
|
|
{
|
|
|
|
|
static std::vector<TestBase *> m_modules_to_test;
|
|
|
|
|
return m_modules_to_test;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void registerTestModule(TestBase *module)
|
|
|
|
|
{
|
|
|
|
|
getTestModules().push_back(module);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// A few item and node definitions for those tests that need them
|
2015-05-05 11:36:40 -04:00
|
|
|
|
extern content_t t_CONTENT_STONE;
|
|
|
|
|
extern content_t t_CONTENT_GRASS;
|
|
|
|
|
extern content_t t_CONTENT_TORCH;
|
|
|
|
|
extern content_t t_CONTENT_WATER;
|
|
|
|
|
extern content_t t_CONTENT_LAVA;
|
|
|
|
|
extern content_t t_CONTENT_BRICK;
|
2015-04-26 01:24:19 -04:00
|
|
|
|
|
2015-10-26 04:13:27 +01:00
|
|
|
|
bool run_tests();
|
2023-06-25 12:13:48 +03:00
|
|
|
|
bool run_tests(const std::string &module_name);
|