| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390 |
- #include <gtest/gtest.h>
- #include <QJsonDocument>
- #include <QJsonObject>
- #include <QJsonArray>
- #include <QTemporaryFile>
- #include "core/entity.h"
- #include "core/world.h"
- #include "core/component.h"
- #include "core/serialization.h"
- #include "systems/nation_id.h"
- #include "units/spawn_type.h"
- #include "units/troop_type.h"
- using namespace Engine::Core;
- class SerializationTest : public ::testing::Test {
- protected:
- void SetUp() override {
- world = std::make_unique<World>();
- }
- void TearDown() override {
- world.reset();
- }
- std::unique_ptr<World> world;
- };
- TEST_F(SerializationTest, EntitySerializationBasic) {
- auto* entity = world->createEntity();
- ASSERT_NE(entity, nullptr);
-
- auto entity_id = entity->getId();
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- EXPECT_TRUE(json.contains("id"));
- EXPECT_EQ(json["id"].toVariant().toULongLong(), static_cast<qulonglong>(entity_id));
- }
- TEST_F(SerializationTest, TransformComponentSerialization) {
- auto* entity = world->createEntity();
- auto* transform = entity->addComponent<TransformComponent>();
-
- transform->position.x = 10.5F;
- transform->position.y = 20.3F;
- transform->position.z = 30.1F;
- transform->rotation.x = 0.5F;
- transform->rotation.y = 1.0F;
- transform->rotation.z = 1.5F;
- transform->scale.x = 2.0F;
- transform->scale.y = 2.5F;
- transform->scale.z = 3.0F;
- transform->hasDesiredYaw = true;
- transform->desiredYaw = 45.0F;
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- ASSERT_TRUE(json.contains("transform"));
- QJsonObject transform_obj = json["transform"].toObject();
-
- EXPECT_FLOAT_EQ(transform_obj["posX"].toDouble(), 10.5);
- EXPECT_FLOAT_EQ(transform_obj["posY"].toDouble(), 20.3);
- EXPECT_FLOAT_EQ(transform_obj["posZ"].toDouble(), 30.1);
- EXPECT_FLOAT_EQ(transform_obj["rotX"].toDouble(), 0.5);
- EXPECT_FLOAT_EQ(transform_obj["rotY"].toDouble(), 1.0);
- EXPECT_FLOAT_EQ(transform_obj["rotZ"].toDouble(), 1.5);
- EXPECT_FLOAT_EQ(transform_obj["scale_x"].toDouble(), 2.0);
- EXPECT_FLOAT_EQ(transform_obj["scaleY"].toDouble(), 2.5);
- EXPECT_FLOAT_EQ(transform_obj["scale_z"].toDouble(), 3.0);
- EXPECT_TRUE(transform_obj["hasDesiredYaw"].toBool());
- EXPECT_FLOAT_EQ(transform_obj["desiredYaw"].toDouble(), 45.0);
- }
- TEST_F(SerializationTest, UnitComponentSerialization) {
- auto* entity = world->createEntity();
- auto* unit = entity->addComponent<UnitComponent>();
-
- unit->health = 80;
- unit->max_health = 100;
- unit->speed = 5.5F;
- unit->vision_range = 15.0F;
- unit->spawn_type = Game::Units::SpawnType::Archer;
- unit->owner_id = 1;
- unit->nation_id = Game::Systems::NationID::RomanRepublic;
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- ASSERT_TRUE(json.contains("unit"));
- QJsonObject unit_obj = json["unit"].toObject();
-
- EXPECT_EQ(unit_obj["health"].toInt(), 80);
- EXPECT_EQ(unit_obj["max_health"].toInt(), 100);
- EXPECT_FLOAT_EQ(unit_obj["speed"].toDouble(), 5.5);
- EXPECT_FLOAT_EQ(unit_obj["vision_range"].toDouble(), 15.0);
- EXPECT_EQ(unit_obj["unit_type"].toString(), QString("archer"));
- EXPECT_EQ(unit_obj["owner_id"].toInt(), 1);
- EXPECT_EQ(unit_obj["nation_id"].toString(), QString("roman_republic"));
- }
- TEST_F(SerializationTest, MovementComponentSerialization) {
- auto* entity = world->createEntity();
- auto* movement = entity->addComponent<MovementComponent>();
-
- movement->hasTarget = true;
- movement->target_x = 50.0F;
- movement->target_y = 60.0F;
- movement->goalX = 55.0F;
- movement->goalY = 65.0F;
- movement->vx = 1.5F;
- movement->vz = 2.0F;
- movement->pathPending = false;
- movement->pendingRequestId = 42;
- movement->repathCooldown = 1.0F;
- movement->lastGoalX = 45.0F;
- movement->lastGoalY = 55.0F;
- movement->timeSinceLastPathRequest = 0.5F;
-
- movement->path.emplace_back(10.0F, 20.0F);
- movement->path.emplace_back(30.0F, 40.0F);
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- ASSERT_TRUE(json.contains("movement"));
- QJsonObject movement_obj = json["movement"].toObject();
-
- EXPECT_TRUE(movement_obj["hasTarget"].toBool());
- EXPECT_FLOAT_EQ(movement_obj["target_x"].toDouble(), 50.0);
- EXPECT_FLOAT_EQ(movement_obj["target_y"].toDouble(), 60.0);
- EXPECT_FLOAT_EQ(movement_obj["goalX"].toDouble(), 55.0);
- EXPECT_FLOAT_EQ(movement_obj["goalY"].toDouble(), 65.0);
- EXPECT_FLOAT_EQ(movement_obj["vx"].toDouble(), 1.5);
- EXPECT_FLOAT_EQ(movement_obj["vz"].toDouble(), 2.0);
- EXPECT_FALSE(movement_obj["pathPending"].toBool());
- EXPECT_EQ(movement_obj["pendingRequestId"].toVariant().toULongLong(), 42ULL);
-
- ASSERT_TRUE(movement_obj.contains("path"));
- QJsonArray path_array = movement_obj["path"].toArray();
- ASSERT_EQ(path_array.size(), 2);
-
- QJsonObject waypoint1 = path_array[0].toObject();
- EXPECT_FLOAT_EQ(waypoint1["x"].toDouble(), 10.0);
- EXPECT_FLOAT_EQ(waypoint1["y"].toDouble(), 20.0);
-
- QJsonObject waypoint2 = path_array[1].toObject();
- EXPECT_FLOAT_EQ(waypoint2["x"].toDouble(), 30.0);
- EXPECT_FLOAT_EQ(waypoint2["y"].toDouble(), 40.0);
- }
- TEST_F(SerializationTest, AttackComponentSerialization) {
- auto* entity = world->createEntity();
- auto* attack = entity->addComponent<AttackComponent>();
-
- attack->range = 10.0F;
- attack->damage = 25;
- attack->cooldown = 2.0F;
- attack->timeSinceLast = 0.5F;
- attack->meleeRange = 2.0F;
- attack->meleeDamage = 15;
- attack->meleeCooldown = 1.5F;
- attack->preferredMode = AttackComponent::CombatMode::Ranged;
- attack->currentMode = AttackComponent::CombatMode::Ranged;
- attack->canMelee = true;
- attack->canRanged = true;
- attack->max_heightDifference = 5.0F;
- attack->inMeleeLock = false;
- attack->meleeLockTargetId = 0;
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- ASSERT_TRUE(json.contains("attack"));
- QJsonObject attack_obj = json["attack"].toObject();
-
- EXPECT_FLOAT_EQ(attack_obj["range"].toDouble(), 10.0);
- EXPECT_EQ(attack_obj["damage"].toInt(), 25);
- EXPECT_FLOAT_EQ(attack_obj["cooldown"].toDouble(), 2.0);
- EXPECT_FLOAT_EQ(attack_obj["timeSinceLast"].toDouble(), 0.5);
- EXPECT_FLOAT_EQ(attack_obj["meleeRange"].toDouble(), 2.0);
- EXPECT_EQ(attack_obj["meleeDamage"].toInt(), 15);
- EXPECT_FLOAT_EQ(attack_obj["meleeCooldown"].toDouble(), 1.5);
- EXPECT_EQ(attack_obj["preferredMode"].toString(), QString("ranged"));
- EXPECT_EQ(attack_obj["currentMode"].toString(), QString("ranged"));
- EXPECT_TRUE(attack_obj["canMelee"].toBool());
- EXPECT_TRUE(attack_obj["canRanged"].toBool());
- EXPECT_FLOAT_EQ(attack_obj["max_heightDifference"].toDouble(), 5.0);
- EXPECT_FALSE(attack_obj["inMeleeLock"].toBool());
- }
- TEST_F(SerializationTest, EntityDeserializationRoundTrip) {
- auto* original_entity = world->createEntity();
- auto* transform = original_entity->addComponent<TransformComponent>();
- transform->position.x = 100.0F;
- transform->position.y = 200.0F;
- transform->position.z = 300.0F;
-
- auto* unit = original_entity->addComponent<UnitComponent>();
- unit->health = 75;
- unit->max_health = 100;
- unit->speed = 6.0F;
-
- QJsonObject json = Serialization::serializeEntity(original_entity);
-
- auto* new_entity = world->createEntity();
- Serialization::deserializeEntity(new_entity, json);
-
- auto* deserialized_transform = new_entity->getComponent<TransformComponent>();
- ASSERT_NE(deserialized_transform, nullptr);
- EXPECT_FLOAT_EQ(deserialized_transform->position.x, 100.0F);
- EXPECT_FLOAT_EQ(deserialized_transform->position.y, 200.0F);
- EXPECT_FLOAT_EQ(deserialized_transform->position.z, 300.0F);
-
- auto* deserialized_unit = new_entity->getComponent<UnitComponent>();
- ASSERT_NE(deserialized_unit, nullptr);
- EXPECT_EQ(deserialized_unit->health, 75);
- EXPECT_EQ(deserialized_unit->max_health, 100);
- EXPECT_FLOAT_EQ(deserialized_unit->speed, 6.0F);
- }
- TEST_F(SerializationTest, DeserializationWithMissingFields) {
- QJsonObject json;
- json["id"] = 1;
-
- QJsonObject unit_obj;
- unit_obj["health"] = 50;
- json["unit"] = unit_obj;
-
- auto* entity = world->createEntity();
- Serialization::deserializeEntity(entity, json);
-
- auto* unit = entity->getComponent<UnitComponent>();
- ASSERT_NE(unit, nullptr);
- EXPECT_EQ(unit->health, 50);
- EXPECT_EQ(unit->max_health, Defaults::kUnitDefaultHealth);
- }
- TEST_F(SerializationTest, DeserializationWithMalformedJSON) {
- QJsonObject json;
- json["id"] = 1;
-
- QJsonObject transform_obj;
- transform_obj["posX"] = "not_a_number";
- json["transform"] = transform_obj;
-
- auto* entity = world->createEntity();
-
- EXPECT_NO_THROW({
- Serialization::deserializeEntity(entity, json);
- });
-
- auto* transform = entity->getComponent<TransformComponent>();
- ASSERT_NE(transform, nullptr);
- EXPECT_FLOAT_EQ(transform->position.x, 0.0F);
- }
- TEST_F(SerializationTest, WorldSerializationRoundTrip) {
- auto* entity1 = world->createEntity();
- auto* transform1 = entity1->addComponent<TransformComponent>();
- transform1->position.x = 10.0F;
-
- auto* entity2 = world->createEntity();
- auto* transform2 = entity2->addComponent<TransformComponent>();
- transform2->position.x = 20.0F;
-
- QJsonDocument doc = Serialization::serializeWorld(world.get());
-
- ASSERT_TRUE(doc.isObject());
- QJsonObject world_obj = doc.object();
- EXPECT_TRUE(world_obj.contains("entities"));
- EXPECT_TRUE(world_obj.contains("nextEntityId"));
- EXPECT_TRUE(world_obj.contains("schemaVersion"));
-
- auto new_world = std::make_unique<World>();
- Serialization::deserializeWorld(new_world.get(), doc);
-
- const auto& entities = new_world->getEntities();
- EXPECT_EQ(entities.size(), 2UL);
- }
- TEST_F(SerializationTest, SaveAndLoadFromFile) {
- auto* entity = world->createEntity();
- auto* transform = entity->addComponent<TransformComponent>();
- transform->position.x = 42.0F;
- transform->position.y = 43.0F;
- transform->position.z = 44.0F;
-
- QJsonDocument doc = Serialization::serializeWorld(world.get());
-
- QTemporaryFile temp_file;
- ASSERT_TRUE(temp_file.open());
- QString filename = temp_file.fileName();
- temp_file.close();
-
- EXPECT_TRUE(Serialization::saveToFile(filename, doc));
-
- QJsonDocument loaded_doc = Serialization::loadFromFile(filename);
- EXPECT_FALSE(loaded_doc.isNull());
-
- auto new_world = std::make_unique<World>();
- Serialization::deserializeWorld(new_world.get(), loaded_doc);
-
- const auto& entities = new_world->getEntities();
- EXPECT_EQ(entities.size(), 1UL);
-
- if (!entities.empty()) {
- auto* loaded_entity = entities.begin()->second.get();
- auto* loaded_transform = loaded_entity->getComponent<TransformComponent>();
- ASSERT_NE(loaded_transform, nullptr);
- EXPECT_FLOAT_EQ(loaded_transform->position.x, 42.0F);
- EXPECT_FLOAT_EQ(loaded_transform->position.y, 43.0F);
- EXPECT_FLOAT_EQ(loaded_transform->position.z, 44.0F);
- }
- }
- TEST_F(SerializationTest, ProductionComponentSerialization) {
- auto* entity = world->createEntity();
- auto* production = entity->addComponent<ProductionComponent>();
-
- production->inProgress = true;
- production->buildTime = 10.0F;
- production->timeRemaining = 5.0F;
- production->producedCount = 3;
- production->maxUnits = 10;
- production->product_type = Game::Units::TroopType::Archer;
- production->rallyX = 100.0F;
- production->rallyZ = 200.0F;
- production->rallySet = true;
- production->villagerCost = 2;
- production->productionQueue.push_back(Game::Units::TroopType::Spearman);
- production->productionQueue.push_back(Game::Units::TroopType::Archer);
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- ASSERT_TRUE(json.contains("production"));
- QJsonObject prod_obj = json["production"].toObject();
-
- EXPECT_TRUE(prod_obj["inProgress"].toBool());
- EXPECT_FLOAT_EQ(prod_obj["buildTime"].toDouble(), 10.0);
- EXPECT_FLOAT_EQ(prod_obj["timeRemaining"].toDouble(), 5.0);
- EXPECT_EQ(prod_obj["producedCount"].toInt(), 3);
- EXPECT_EQ(prod_obj["maxUnits"].toInt(), 10);
- EXPECT_EQ(prod_obj["product_type"].toString(), QString("archer"));
- EXPECT_FLOAT_EQ(prod_obj["rallyX"].toDouble(), 100.0);
- EXPECT_FLOAT_EQ(prod_obj["rallyZ"].toDouble(), 200.0);
- EXPECT_TRUE(prod_obj["rallySet"].toBool());
- EXPECT_EQ(prod_obj["villagerCost"].toInt(), 2);
-
- ASSERT_TRUE(prod_obj.contains("queue"));
- QJsonArray queue = prod_obj["queue"].toArray();
- EXPECT_EQ(queue.size(), 2);
- EXPECT_EQ(queue[0].toString(), QString("spearman"));
- EXPECT_EQ(queue[1].toString(), QString("archer"));
- }
- TEST_F(SerializationTest, PatrolComponentSerialization) {
- auto* entity = world->createEntity();
- auto* patrol = entity->addComponent<PatrolComponent>();
-
- patrol->currentWaypoint = 1;
- patrol->patrolling = true;
- patrol->waypoints.emplace_back(10.0F, 20.0F);
- patrol->waypoints.emplace_back(30.0F, 40.0F);
- patrol->waypoints.emplace_back(50.0F, 60.0F);
-
- QJsonObject json = Serialization::serializeEntity(entity);
-
- ASSERT_TRUE(json.contains("patrol"));
- QJsonObject patrol_obj = json["patrol"].toObject();
-
- EXPECT_EQ(patrol_obj["currentWaypoint"].toInt(), 1);
- EXPECT_TRUE(patrol_obj["patrolling"].toBool());
-
- ASSERT_TRUE(patrol_obj.contains("waypoints"));
- QJsonArray waypoints = patrol_obj["waypoints"].toArray();
- EXPECT_EQ(waypoints.size(), 3);
-
- QJsonObject wp0 = waypoints[0].toObject();
- EXPECT_FLOAT_EQ(wp0["x"].toDouble(), 10.0);
- EXPECT_FLOAT_EQ(wp0["y"].toDouble(), 20.0);
- }
- TEST_F(SerializationTest, EmptyWorldSerialization) {
- QJsonDocument doc = Serialization::serializeWorld(world.get());
-
- ASSERT_TRUE(doc.isObject());
- QJsonObject world_obj = doc.object();
-
- EXPECT_TRUE(world_obj.contains("entities"));
- QJsonArray entities = world_obj["entities"].toArray();
- EXPECT_EQ(entities.size(), 0);
- }
|