DarkflameServer/tests/dECSTests/TestECS.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

221 lines
5.7 KiB
C++
Raw Normal View History

2024-12-18 04:06:41 +00:00
#include <cstdint>
#include <cstddef>
#include <span>
#include <optional>
#include <gtest/gtest.h>
2024-12-17 06:13:14 +00:00
#include "Core.h"
#include <dComponents/Component.h>
#include <eReplicaComponentType.h>
2024-12-17 06:13:14 +00:00
using namespace dECS;
2024-12-18 04:06:41 +00:00
namespace TestECS {
using LegacyComponent = ::Component;
2024-12-17 06:13:14 +00:00
2024-12-18 04:06:41 +00:00
namespace Component {
using enum eReplicaComponentType;
void* NULL_PARENT = nullptr;
struct Legacy : public LegacyComponent {
static constexpr eReplicaComponentType ComponentType = CHANGLING;
Legacy() = default;
void Update(float deltaTime) {
std::printf("Legacy updated!\n");
}
};
struct Invalid {
static constexpr eReplicaComponentType ComponentType = INVALID;
int value;
};
struct Destroyable {
static constexpr eReplicaComponentType ComponentType = DESTROYABLE;
using FactionId = int32_t;
float health;
float maxHealth;
float armor;
float maxArmor;
float imag;
float maxImag;
uint32_t damageToAbsorb;
bool immune;
bool gmImmune;
bool shielded;
float actualMaxHealth;
float actualMaxArmor;
float actualMaxImagination;
std::vector<FactionId> factionIds;
bool smashable;
};
}
struct IFakeSystem {
virtual ~IFakeSystem() = default;
[[nodiscard]]
constexpr virtual size_t Count() const noexcept = 0;
constexpr virtual void EmplaceBack() = 0;
};
template <typename... Cs>
struct FakeSystem : public IFakeSystem {
template <typename C>
using Storage = std::vector<std::remove_const_t<C>>;
std::tuple<Storage<Cs>...> data;
[[nodiscard]]
constexpr size_t Count() const noexcept override {
return std::get<0>(data).size();
}
constexpr void EmplaceBack() override {
(std::get<Storage<Cs>>(data).emplace_back(), ...);
}
template <typename C>
requires std::disjunction_v<std::is_same<C, Cs>...>
[[nodiscard]]
std::span<C> Get() {
return std::get<Storage<C>>(data);
}
template <typename C>
requires std::disjunction_v<std::is_same<C, Cs>...>
[[nodiscard]]
std::span<const C> Get() const {
return std::get<Storage<C>>(data);
}
template <typename Fn>
requires std::is_invocable_r_v<void, Fn(Cs...), Cs...>
void ForEach(Fn&& fn);
};
class FakeIter {
public:
constexpr FakeIter(const IFakeSystem& fakeSys) noexcept
: m_System{ fakeSys }
{}
[[nodiscard]]
constexpr bool Next() {
return m_Count++ > m_System.Count();
}
private:
size_t m_Count;
const IFakeSystem& m_System;
};
template <typename... Cs>
template <typename Fn>
requires std::is_invocable_r_v<void, Fn(Cs...), Cs...>
void FakeSystem<Cs...>::ForEach(Fn&& fn){
for (size_t i = 0; i < Count(); ++i) {
fn(Get<Cs>()[i]...);
}
}
}
2024-12-17 06:13:14 +00:00
// Test that entity IDs increment correctly
TEST(ECSTest, IncrementEntityIdsSingleThread) {
auto w = World{};
auto ea = w.MakeEntity();
ASSERT_EQ(ea.GetObjectID(), 1);
2024-12-17 06:13:14 +00:00
auto eb = w.MakeEntity();
ASSERT_EQ(eb.GetObjectID(), 2);
2024-12-17 06:13:14 +00:00
auto ec = w.MakeEntity();
ASSERT_EQ(ec.GetObjectID(), 3);
2024-12-17 06:13:14 +00:00
}
// Test adding and getting components
TEST(ECSTest, MakeOneEntityAndAddComponents) {
2024-12-18 04:06:41 +00:00
using namespace TestECS::Component;
2024-12-17 06:13:14 +00:00
auto w = World{};
auto e = w.MakeEntity();
ASSERT_EQ(e.GetObjectID(), 1);
2024-12-17 06:13:14 +00:00
// add component
2024-12-18 04:06:41 +00:00
auto* const testCompPtr = e.AddComponent<Invalid>();
2024-12-17 06:13:14 +00:00
ASSERT_NE(testCompPtr, nullptr);
2024-12-18 04:06:41 +00:00
ASSERT_EQ(testCompPtr->ComponentType, Invalid::ComponentType);
2024-12-17 06:13:14 +00:00
ASSERT_EQ(testCompPtr->value, 0);
testCompPtr->value = 15;
// try getting the same component we just added
2024-12-18 04:06:41 +00:00
auto* const gotTestCompPtr = e.GetComponent<Invalid>();
2024-12-17 07:02:30 +00:00
ASSERT_NE(gotTestCompPtr, nullptr);
ASSERT_EQ(gotTestCompPtr, testCompPtr);
ASSERT_NE(gotTestCompPtr->value, 0);
ASSERT_EQ(gotTestCompPtr->value, 15);
2024-12-17 06:13:14 +00:00
}
// Test world scoping
TEST(ECSTest, WorldScope) {
2024-12-18 04:06:41 +00:00
using namespace TestECS::Component;
2024-12-17 06:13:14 +00:00
auto e = std::optional<dECS::Entity>{};
{
auto w = World{};
e.emplace(w.MakeEntity());
ASSERT_EQ(e->GetObjectID(), 1);
2024-12-17 06:13:14 +00:00
// add component within scope
2024-12-18 04:06:41 +00:00
auto* const cPtr = e->AddComponent<Invalid>();
2024-12-17 06:13:14 +00:00
ASSERT_NE(cPtr, nullptr);
}
2024-12-17 06:25:29 +00:00
// Attempting to access this component should return nullptr
// now that the world has gone out of scope
2024-12-18 04:06:41 +00:00
ASSERT_EQ(e->GetComponent<Invalid>(), nullptr);
}
// Create and iterate over a system
TEST(ECSTest, CreateAndIterateOverSystem) {
using namespace TestECS::Component;
auto w = World{};
auto s = w.MakeSystem<Destroyable, const Invalid>("DestInvalid");
size_t count = 0;
s.ForEach([&](Destroyable& d, const Invalid& i) {
std::printf("i = %ld: d.health = %f\n", ++count, d.health);
d.health += 1;
});
}
TEST(ECSTest, FakeIterationForTestingPurposes) {
using namespace TestECS;
using namespace TestECS::Component;
auto s = FakeSystem<Legacy, Destroyable>{};
auto const r = 2 + std::rand() % 8;
for (size_t i = 0; i < r; ++i) {
s.EmplaceBack();
}
size_t count = 0;
s.ForEach([&](Legacy& l, Destroyable& d) {
l.Update(0.0f);
std::printf("i = %ld: d.health = %f\n", ++count, d.health);
d.health += 1;
});
std::printf("Total count = %ld\n", count);
ASSERT_EQ(r, count);
}