From 545db046d79283e995f37f83504e08d39f606f30 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Sat, 26 Oct 2024 23:14:51 +0300 Subject: [PATCH 01/12] Feature - extend checking building and research requisities from queues --- app/Services/BuildingQueueService.php | 108 ++++++++++++----- app/Services/ObjectService.php | 22 +++- app/Services/PlanetService.php | 18 +++ app/Services/PlayerService.php | 11 ++ app/Services/ResearchQueueService.php | 150 ++++++++++++++++-------- app/Services/UnitQueueService.php | 6 +- tests/Unit/BuildingQueueServiceTest.php | 89 ++++++++++++++ tests/Unit/ObjectServiceTest.php | 77 ++++++++++-- tests/Unit/PlanetServiceTest.php | 25 ++++ tests/Unit/ResearchQueueServiceTest.php | 84 +++++++++++++ 10 files changed, 499 insertions(+), 91 deletions(-) create mode 100644 tests/Unit/BuildingQueueServiceTest.php create mode 100644 tests/Unit/ResearchQueueServiceTest.php diff --git a/app/Services/BuildingQueueService.php b/app/Services/BuildingQueueService.php index 64b328b6..40c32bf0 100644 --- a/app/Services/BuildingQueueService.php +++ b/app/Services/BuildingQueueService.php @@ -56,6 +56,23 @@ public function retrieveFinished(int $planet_id): Collection ->get(); } + /** + * Get building queue items + * + * @return Collection + */ + public function retrieveQueueItems(PlanetService $planet): Collection + { + // Fetch queue items from model + return BuildingQueue::where([ + ['planet_id', $planet->getPlanetId()], + ['processed', 0], + ['canceled', 0], + ]) + ->orderBy('time_start', 'asc') + ->get(); + } + /** * Add a building to the building queue for the current planet. * @@ -113,14 +130,7 @@ public function add(PlanetService $planet, int $building_id): void */ public function retrieveQueue(PlanetService $planet): BuildingQueueListViewModel { - // Fetch queue items from model - $queue_items = BuildingQueue::where([ - ['planet_id', $planet->getPlanetId()], - ['processed', 0], - ['canceled', 0], - ]) - ->orderBy('time_start', 'asc') - ->get(); + $queue_items = $this->retrieveQueueItems($planet); // Convert to ViewModel array $list = array(); @@ -231,6 +241,14 @@ public function start(PlanetService $planet, int $time_start = 0): void continue; } + // Sanity check: check if the building requirements are still met. If not, + // then cancel build request. + if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer(), $queue_item->object_level_target)) { + $this->cancel($planet, $queue_item->id, $queue_item->object_id); + + continue; + } + // All OK, deduct resources and start building process. $planet->deductResources($price); @@ -277,26 +295,6 @@ public function cancel(PlanetService $planet, int $building_queue_id, int $build // If object is found: add canceled flag. if ($queue_item) { - // Gets all building queue records of this target level and all that - // come after it. So e.g. if user cancels build order for metal mine - // level 5 then any other already queued build orders for lvl 6,7,8 etc. - // will also be canceled. - $queue_items_higher_level = BuildingQueue::where([ - ['planet_id', $planet->getPlanetId()], - ['object_id', $building_id], - ['object_level_target', '>', $queue_item->object_level_target], - ['processed', 0], - ['canceled', 0], - ])->get(); - - // Add canceled flag to all entries with a higher level (if any). - foreach ($queue_items_higher_level as $queue_item_higher_level) { - $queue_item_higher_level->building = 0; - $queue_item_higher_level->canceled = 1; - - $queue_item_higher_level->save(); - } - // Give back resources if the current entry was already building. if ($queue_item->building === 1) { $planet->addResources(new Resources($queue_item->metal, $queue_item->crystal, $queue_item->deuterium, 0)); @@ -308,8 +306,62 @@ public function cancel(PlanetService $planet, int $building_queue_id, int $build $queue_item->save(); + // Check if requirements for all other items in the queue are still met. + // So e.g. if user cancels build order for metal mine + // level 5 then any other already queued build orders for lvl 6,7,8 etc. + // will also be canceled. Same applies to building requirements, + // if user cancels build order for robotics factory which is requirement + // for shipyard then shipyard will also be canceled. + // Requirements are checked only for building queue objects as + // unit queue objects cannot be canceled. + $this->cancelItemMissingRequirements($planet); + + $research_queue = resolve('OGame\Services\ResearchQueueService'); + $research_queue->cancelItemMissingRequirements($planet->getPlayer(), $planet); + // Set the next queue item to start (if applicable) $this->start($planet); } } + + /** + * Get is object in building queue + * + * @return bool + */ + public function objectInBuildingQueue(PlanetService $planet, string $machine_name, int $level): bool + { + $queue_items = $this->retrieveQueueItems($planet); + + foreach ($queue_items as $item) { + $object = $this->objects->getObjectById($item->object_id); + + if($object->machine_name === $machine_name && $item->object_level_target === $level) { + return true; + } + } + + return false; + } + + /** + * Cancel first building queue item missing requirements. + * This function will be called recursively when it cancels the item. + * + * @return void + */ + public function cancelItemMissingRequirements(PlanetService $planet): void + { + $build_queue_items = $this->retrieveQueueItems($planet); + + foreach ($build_queue_items as $build_queue_item) { + $object = $this->objects->getObjectById($build_queue_item->object_id); + + if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer(), $build_queue_item->object_level_target)) { + $this->cancel($planet, $build_queue_item->id, $object->id); + break; + } + + } + } } diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index cbe710ad..0c31019b 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -329,21 +329,37 @@ public function getBuildingObjectsWithStorage(): array * @param string $machine_name * @param PlanetService $planet * @param PlayerService $player + * @param int $level * @return bool */ - public function objectRequirementsMet(string $machine_name, PlanetService $planet, PlayerService $player): bool + public function objectRequirementsMet(string $machine_name, PlanetService $planet, PlayerService $player, int $level = 0): bool { try { $object = $this->getObjectByMachineName($machine_name); + + // Check if object prior levels are built or are in build queue + if($level) { + $current_level = $planet->getObjectLevel($object->machine_name); + + // Check missing levels from build queue + for($i = $current_level + 1; $i < $level; $i++) { + if(!$planet->isBuildingObject($object->machine_name, $i)) { + return false; + } + } + } + foreach ($object->requirements as $requirement) { // Load required object and check if requirements are met. $object_required = $this->getObjectByMachineName($requirement->object_machine_name); if ($object_required->type === GameObjectType::Research) { - if ($player->getResearchLevel($object_required->machine_name) < $requirement->level) { + if ($player->getResearchLevel($object_required->machine_name) < $requirement->level && !$player->isResearchingTech($requirement->object_machine_name, $requirement->level)) { return false; } } else { - if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level) { + // Check if requirements are met with existing buildings or with buildings on build queue. + // Building queue is checked only for building queue objects, not for unit queue objects. + if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level && !$planet->isBuildingObject($requirement->object_machine_name, $requirement->level)) { return false; } } diff --git a/app/Services/PlanetService.php b/app/Services/PlanetService.php index ecc9ccea..33e93619 100644 --- a/app/Services/PlanetService.php +++ b/app/Services/PlanetService.php @@ -1459,6 +1459,24 @@ public function isBuilding(): bool return count($build_queue) > 0; } + /** + * Get is the current planet building the object or not + * + * @return bool + */ + public function isBuildingObject(string $machine_name, int $level): bool + { + $object = $this->objects->getObjectByMachineName($machine_name); + + // Check only building queue objects + if($object->type !== GameObjectType::Building && $object->type !== GameObjectType::Station) { + return false; + } + + $build_queue = resolve(BuildingQueueService::class); + return $build_queue->objectInBuildingQueue($this, $machine_name, $level); + } + /** * Get building count from planet * diff --git a/app/Services/PlayerService.php b/app/Services/PlayerService.php index 1b97ccbf..776b4505 100644 --- a/app/Services/PlayerService.php +++ b/app/Services/PlayerService.php @@ -572,4 +572,15 @@ public function delete(): void // Delete the actual user. $this->user->delete(); } + + /** + * Get is the player researching the tech or not + * + * @return bool + */ + public function isResearchingTech(string $machine_name, int $level): bool + { + $research_queue = resolve('OGame\Services\ResearchQueueService'); + return $research_queue->objectInResearchQueue($this, $machine_name, $level); + } } diff --git a/app/Services/ResearchQueueService.php b/app/Services/ResearchQueueService.php index d71c9f52..9e6f17da 100644 --- a/app/Services/ResearchQueueService.php +++ b/app/Services/ResearchQueueService.php @@ -105,6 +105,27 @@ public function retrieveFinishedForUser(PlayerService $player): \Illuminate\Supp ->get(); } + /** + * Get research queue items + * + * @return Collection + */ + public function retrieveQueueItems(PlayerService $player): Collection + { + // Fetch queue items from model + return $this->model + ->join('planets', 'research_queues.planet_id', '=', 'planets.id') + ->join('users', 'planets.user_id', '=', 'users.id') + ->where([ + ['users.id', $player->getId()], + ['research_queues.processed', 0], + ['research_queues.canceled', 0], + ]) + ->select('research_queues.*') + ->orderBy('research_queues.time_start', 'asc') + ->get(); + } + /** * Add a research object to the research queue for the current planet. * @@ -148,7 +169,7 @@ public function add(PlayerService $player, PlanetService $planet, int $research_ } /** - * Retrieve current building build queue for a planet. + * Retrieve current research queue for a planet. * * @param PlanetService $planet * @return ResearchQueueListViewModel @@ -219,9 +240,9 @@ public function activeBuildingQueueItemCount(PlayerService $player, int $buildin } /** - * Start building the next item in the queue (if available). + * Start researching the next item in the queue (if available). * - * This actually starts the building process and deducts the resources + * This actually starts the research process and deducts the resources * from the planet. If there are not enough resources the build attempt * will fail. * @@ -229,7 +250,7 @@ public function activeBuildingQueueItemCount(PlayerService $player, int $buildin * * @param int $time_start * Optional parameter to indicate when the new item should start, this - * is used for when a few build queue items are finished at the exact + * is used for when a few research queue items are finished at the exact * same time, e.g. when a user closes its session and logs back in * after a while. * @return void @@ -253,17 +274,17 @@ public function start(PlayerService $player, int $time_start = 0): void $planet = $player->planets->childPlanetById($queue_item->planet_id); $object = $this->objects->getResearchObjectById($queue_item->object_id); - // See if the planet has enough resources for this build attempt. + // See if the planet has enough resources for this research attempt. $price = $this->objects->getObjectPrice($object->machine_name, $planet); - $build_time = $player->planets->current()->getTechnologyResearchTime($object->machine_name); + $research_time = $player->planets->current()->getTechnologyResearchTime($object->machine_name); - // Only start the queue item if there are no other queue items building + // Only start the queue item if there are no other queue items researching // for this planet. - $build_queue = $this->retrieveQueue($planet); - $currently_building = $build_queue->getCurrentlyBuildingFromQueue(); + $research_queue = $this->retrieveQueue($planet); + $currently_researching = $research_queue->getCurrentlyBuildingFromQueue(); - if ($currently_building !== null) { - // There already is something else building, don't start a new one. + if ($currently_researching !== null) { + // There already is something else researching, don't start a new one. break; } @@ -272,29 +293,37 @@ public function start(PlayerService $player, int $time_start = 0): void // is wrong. $current_level = $player->getResearchLevel($object->machine_name); if ($queue_item->object_level_target !== ($current_level + 1)) { - // Error, cancel build queue item. + // Error, cancel research queue item. $this->cancel($player, $queue_item->id, $queue_item->object_id); continue; } // Sanity check: check if the planet has enough resources. If not, - // then cancel build request. + // then cancel research request. if (!$planet->hasResources($price)) { - // Error, cancel build queue item. + // Error, cancel research queue item. $this->cancel($player, $queue_item->id, $queue_item->object_id); continue; } - // All OK, deduct resources and start building process. + // Sanity check: check if the researching requirements are still met. If not, + // then cancel research request. + if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $player, $queue_item->object_level_target)) { + $this->cancel($player, $queue_item->id, $queue_item->object_id); + + continue; + } + + // All OK, deduct resources and start researching process. $planet->deductResources($price); if (!$time_start) { $time_start = (int)Carbon::now()->timestamp; } - $queue_item->time_duration = (int)$build_time; + $queue_item->time_duration = (int)$research_time; $queue_item->time_start = $time_start; $queue_item->time_end = $queue_item->time_start + $queue_item->time_duration; $queue_item->building = 1; @@ -305,7 +334,7 @@ public function start(PlayerService $player, int $time_start = 0): void // If the calculated end time is lower than the current time, // we force that the planet is updated again which will grant - // the building immediately without having to wait for a refresh. + // the research immediately without having to wait for a refresh. if ($queue_item->time_end < Carbon::now()->timestamp) { $player->updateResearchQueue(); } @@ -313,22 +342,22 @@ public function start(PlayerService $player, int $time_start = 0): void } /** - * Cancels an active building queue record. + * Cancels an active research queue record. * * @param PlayerService $player - * @param int $building_queue_id - * @param int $building_id + * @param int $research_queue_id + * @param int $research_id * @throws Exception */ - public function cancel(PlayerService $player, int $building_queue_id, int $building_id): void + public function cancel(PlayerService $player, int $research_queue_id, int $research_id): void { $queue_item = $this->model ->join('planets', 'research_queues.planet_id', '=', 'planets.id') ->join('users', 'planets.user_id', '=', 'users.id') ->where([ ['users.id', $player->getId()], - ['research_queues.id', $building_queue_id], - ['object_id', $building_id], + ['research_queues.id', $research_queue_id], + ['object_id', $research_id], ['processed', 0], ['canceled', 0], ]) @@ -339,36 +368,18 @@ public function cancel(PlayerService $player, int $building_queue_id, int $build if ($queue_item) { // Typecast to a new object to avoid issues with the model. $queue_item = $queue_item instanceof ResearchQueue ? $queue_item : new ResearchQueue($queue_item->getAttributes()); - $planetService = $player->planets->childPlanetById($queue_item->planet_id); + $planet = $player->planets->childPlanetById($queue_item->planet_id); - // Gets all building queue records of this target level and all that + // Gets all research queue records of this target level and all that // come after it. So e.g. if user cancels build order for metal mine // level 5 then any other already queued build orders for lvl 6,7,8 etc. - // will also be canceled. - $queue_items_higher_level = $this->model - ->join('planets', 'research_queues.planet_id', '=', 'planets.id') - ->join('users', 'planets.user_id', '=', 'users.id') - ->where([ - ['users.id', $player->getId()], - ['object_id', $building_id], - ['object_level_target', '>', $queue_item->object_level_target], - ['processed', 0], - ['canceled', 0], - ]) - ->select('research_queues.*') - ->get(); - - // Add canceled flag to all entries with a higher level (if any). - foreach ($queue_items_higher_level as $queue_item_higher_level) { - $queue_item_higher_level->building = 0; - $queue_item_higher_level->canceled = 1; - - $queue_item_higher_level->save(); - } + // will also be canceled. Same applies to research requirements, + // if user cancels research order for Energy Technology which is requirement + // for Impulse Drive then Impulse Drive will also be canceled. // Give back resources if the current entry was already building. if ($queue_item->building === 1) { - $planetService->addResources(new Resources($queue_item->metal, $queue_item->crystal, $queue_item->deuterium, 0)); + $planet->addResources(new Resources($queue_item->metal, $queue_item->crystal, $queue_item->deuterium, 0)); } // Add canceled flag to the main entry. @@ -377,8 +388,53 @@ public function cancel(PlayerService $player, int $building_queue_id, int $build $queue_item->save(); + $this->cancelItemMissingRequirements($player, $planet); + + $build_queue = resolve('OGame\Services\BuildingQueueService'); + $build_queue->cancelItemMissingRequirements($planet); + // Set the next queue item to start (if applicable) $this->start($player); } } + + /** + * Get is object in research queue + * + * @return bool + */ + public function objectInResearchQueue(PlayerService $player, string $machine_name, int $level): bool + { + $queue_items = $this->retrieveQueueItems($player); + + foreach ($queue_items as $item) { + $object = $this->objects->getObjectById($item->object_id); + + if($object->machine_name === $machine_name && $item->object_level_target === $level) { + return true; + } + } + + return false; + } + + /** + * Cancel first research queue item missing requirements. + * This function will be called recursively when it cancels the item. + * + * @return void + */ + public function cancelItemMissingRequirements(PlayerService $player, PlanetService $planet): void + { + $research_queue_items = $this->retrieveQueueItems($player); + + foreach ($research_queue_items as $research_queue_item) { + $object = $this->objects->getObjectById($research_queue_item->object_id); + + if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $player, $research_queue_item->object_level_target)) { + $this->cancel($player, $research_queue_item->id, $object->id); + break; + } + } + } } diff --git a/app/Services/UnitQueueService.php b/app/Services/UnitQueueService.php index 7b42f711..62918e93 100644 --- a/app/Services/UnitQueueService.php +++ b/app/Services/UnitQueueService.php @@ -34,7 +34,7 @@ class UnitQueueService private UnitQueue $model; /** - * BuildingQueue constructor. + * UnitQueueService constructor. */ public function __construct(ObjectService $objects) { @@ -44,7 +44,7 @@ public function __construct(ObjectService $objects) } /** - * Retrieve current building build queue for a planet. + * Retrieve current unit build queue for a planet. * * @param PlanetService $planet * @return UnitQueueListViewModel @@ -100,7 +100,7 @@ public function retrieveQueue(PlanetService $planet): UnitQueueListViewModel } /** - * Retrieve current building build queue for a planet. + * Retrieve current unit build queue for a planet. * * @param int $planet_id * @return Collection diff --git a/tests/Unit/BuildingQueueServiceTest.php b/tests/Unit/BuildingQueueServiceTest.php new file mode 100644 index 00000000..075b97ec --- /dev/null +++ b/tests/Unit/BuildingQueueServiceTest.php @@ -0,0 +1,89 @@ +building_queue = new BuildingQueueService($object_service); + + $planet = Planet::factory()->make(['id' => 1]); + $this->planetService->setPlanet($planet); + + $this->player = User::factory()->make(['id' => 1]); + $this->playerService->load(1); + } + + /** + * Tests object is found from building queue + */ + public function testIsObjectInBuildingQueue(): void + { + $planet = Planet::factory()->make(['id' => 1]); + $this->planetService->setPlanet($planet); + + // Add level 3 shipyard to building queue + $queue = new BuildingQueue(); + $queue->planet_id = $this->planetService->getPlanetId(); + $queue->object_id = 21; + $queue->object_level_target = 3; + $queue->save(); + + $this->assertTrue($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 3)); + $this->assertFalse($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 4)); + $this->assertFalse($this->building_queue->objectInBuildingQueue($this->planetService, 'robot_factory', 3)); + } + + /** + * Tests building queue item is cancelled if requirements are not met. + */ + public function testCancelItemMissingRequirements(): void + { + // Add level 2 robot factory to building queue + $queue_robot_factory = new BuildingQueue(); + $queue_robot_factory->planet_id = $this->planetService->getPlanetId(); + $queue_robot_factory->object_id = 14; + $queue_robot_factory->object_level_target = 2; + $queue_robot_factory->save(); + + // Add level 1 shipyard to building queue + $queue = new BuildingQueue(); + $queue->planet_id = $this->planetService->getPlanetId(); + $queue->object_id = 21; + $queue->object_level_target = 1; + $queue->save(); + + // Assert that shipyard is in building queue + $this->assertTrue($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 1)); + + // Cancel robot factory + $this->building_queue->cancel($this->planetService, $queue_robot_factory->id, 14); + $this->building_queue->cancelItemMissingRequirements($this->planetService); + + // Assert that shipyard is in building queue + $this->assertFalse($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 1)); + } +} diff --git a/tests/Unit/ObjectServiceTest.php b/tests/Unit/ObjectServiceTest.php index 63cb69ab..fd5e5903 100644 --- a/tests/Unit/ObjectServiceTest.php +++ b/tests/Unit/ObjectServiceTest.php @@ -2,31 +2,57 @@ namespace Tests\Unit; +use Illuminate\Foundation\Testing\DatabaseTransactions; +use OGame\Models\BuildingQueue; +use OGame\Models\Planet; use OGame\Services\ObjectService; use Tests\UnitTestCase; class ObjectServiceTest extends UnitTestCase { + use DatabaseTransactions; + + protected ObjectService $object_service; + + protected Planet $planet; + + /** + * Set up common test components. + * + * @throws BindingResolutionException + */ + protected function setUp(): void + { + parent::setUp(); + + $this->object_service = new ObjectService(); + + $this->planet = Planet::factory()->make(['id' => 1]); + $this->planetService->setPlanet($this->planet); + } + + /** + * Tests maximum building amount returns correct value. + */ public function testGetObjectMaxBuildAmount(): void { - $objectService = new ObjectService(); $this->createAndSetPlanetModel([]); - // Test with requirement not met - $maxBuildAmount = $objectService->getObjectMaxBuildAmount('plasma_turret', $this->planetService, false); - $this->assertEquals(0, $maxBuildAmount); + // Test with requirements not met + $max_build_amount = $this->object_service->getObjectMaxBuildAmount('plasma_turret', $this->planetService, false); + $this->assertEquals(0, $max_build_amount); // Test with object limited to one instance - $maxBuildAmount = $objectService->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); - $this->assertEquals(1, $maxBuildAmount); + $max_build_amount = $this->object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); + $this->assertEquals(1, $max_build_amount); $this->createAndSetPlanetModel([ 'small_shield_dome' => 1, ]); // Test with object limited to one instance which already exists - $maxBuildAmount = $objectService->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); - $this->assertEquals(0, $maxBuildAmount); + $max_build_amount = $this->object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); + $this->assertEquals(0, $max_build_amount); $this->createAndSetPlanetModel([ 'metal' => 24000, @@ -34,7 +60,38 @@ public function testGetObjectMaxBuildAmount(): void ]); // Test it calculates max amount correctly - $maxBuildAmount = $objectService->getObjectMaxBuildAmount('anti_ballistic_missile', $this->planetService, true); - $this->assertEquals(3, $maxBuildAmount); + $max_build_amount = $this->object_service->getObjectMaxBuildAmount('anti_ballistic_missile', $this->planetService, true); + $this->assertEquals(3, $max_build_amount); + } + + /** + * Tests object requirements are verified against prior levels, research and buildings + * including items in building and research queues. + */ + public function testObjectRequirementsMet(): void + { + // Assert that requirements are not met if prior levels doesn't exist + $this->assertFalse($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->playerService, 2)); + + $this->planet->robot_factory = 1; + + // Assert that requirements are met if prior levels exists + $this->assertTrue($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->playerService, 2)); + + // Assert that requirements are not met if requisites are missing + $this->assertFalse($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->playerService, 1)); + + // Add shipyard to build queue + $queue = new BuildingQueue(); + $queue->planet_id = $this->planetService->getPlanetId(); + $queue->object_id = 21; + $queue->object_level_target = 1; + $queue->save(); + + // Assert that requirements are met if requisites are in build queue + $this->assertTrue($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->playerService, 1)); + + // Assert that research requirements are not met if building requirements are not met + $this->assertFalse($this->object_service->objectRequirementsMet('computer_technology', $this->planetService, $this->playerService, 1)); } } diff --git a/tests/Unit/PlanetServiceTest.php b/tests/Unit/PlanetServiceTest.php index d6b2691c..138c0aab 100644 --- a/tests/Unit/PlanetServiceTest.php +++ b/tests/Unit/PlanetServiceTest.php @@ -3,12 +3,17 @@ namespace Tests\Unit; use Illuminate\Contracts\Container\BindingResolutionException; +use Illuminate\Foundation\Testing\DatabaseTransactions; +use OGame\Models\BuildingQueue; use OGame\Models\Enums\ResourceType; +use OGame\Models\Planet; use OGame\Models\Resources; use Tests\UnitTestCase; class PlanetServiceTest extends UnitTestCase { + use DatabaseTransactions; + /** * Set up common test components. * @@ -192,4 +197,24 @@ public function testGetPlanetBuildingCount(): void $this->assertEquals(150, $this->planetService->getBuildingCount()); } + + /** + * Tests object building queue status. + */ + public function testIsBuildingObject(): void + { + $planet = Planet::factory()->make(['id' => 1]); + $this->planetService->setPlanet($planet); + + // Add level 3 shipyard to building queue + $queue = new BuildingQueue(); + $queue->planet_id = $this->planetService->getPlanetId(); + $queue->object_id = 21; + $queue->object_level_target = 3; + $queue->save(); + + $this->assertTrue($this->planetService->isBuildingObject('shipyard', 3)); + $this->assertFalse($this->planetService->isBuildingObject('shipyard', 4)); + $this->assertFalse($this->planetService->isBuildingObject('robot_factory', 3)); + } } diff --git a/tests/Unit/ResearchQueueServiceTest.php b/tests/Unit/ResearchQueueServiceTest.php new file mode 100644 index 00000000..cf2a346b --- /dev/null +++ b/tests/Unit/ResearchQueueServiceTest.php @@ -0,0 +1,84 @@ +research_queue = new ResearchQueueService($object_service); + + $planet = Planet::factory()->make(['id' => 1]); + $this->planetService->setPlanet($planet); + + User::factory()->make(['id' => 1]); + $this->playerService->load(1); + } + + /** + * Tests object is found from research queue + */ + public function testIsObjectInResearchQueue(): void + { + // Add level 3 impulse drive to research queue + $queue = new ResearchQueue(); + $queue->planet_id = $this->planetService->getPlanetId(); + $queue->object_id = 117; + $queue->object_level_target = 3; + $queue->save(); + + $this->assertTrue($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 3)); + $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 4)); + $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'energy_technology', 4)); + } + + /** + * Tests research queue item is cancelled if requirements are not met. + */ + public function testCancelItemMissingRequirements(): void + { + // Add level 1 energy technology to research queue + $queue_energy_tech = new ResearchQueue(); + $queue_energy_tech->planet_id = $this->planetService->getPlanetId(); + $queue_energy_tech->object_id = 113; + $queue_energy_tech->object_level_target = 1; + $queue_energy_tech->save(); + + // Add level 1 impulse drive to research queue + $queue = new ResearchQueue(); + $queue->planet_id = $this->planetService->getPlanetId(); + $queue->object_id = 117; + $queue->object_level_target = 1; + $queue->save(); + + // Assert that impulse drive is in research queue + $this->assertTrue($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 1)); + + // Cancel energy technology + $this->research_queue->cancel($this->playerService, $queue_energy_tech->id, 113); + $this->research_queue->cancelItemMissingRequirements($this->playerService, $this->planetService); + + // Assert that impulse drive is in research queue + $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 1)); + } +} From 2fb28e6ed356ed7f50732cd00830ce9c7574c0ec Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Sat, 26 Oct 2024 23:22:35 +0300 Subject: [PATCH 02/12] Code cleanup --- app/Services/BuildingQueueService.php | 2 +- app/Services/ObjectService.php | 6 +++--- app/Services/PlanetService.php | 4 ++-- app/Services/PlayerService.php | 2 +- app/Services/ResearchQueueService.php | 4 ++-- tests/Unit/BuildingQueueServiceTest.php | 2 +- tests/Unit/ResearchQueueServiceTest.php | 4 ++-- 7 files changed, 12 insertions(+), 12 deletions(-) diff --git a/app/Services/BuildingQueueService.php b/app/Services/BuildingQueueService.php index 40c32bf0..81748bf2 100644 --- a/app/Services/BuildingQueueService.php +++ b/app/Services/BuildingQueueService.php @@ -336,7 +336,7 @@ public function objectInBuildingQueue(PlanetService $planet, string $machine_nam foreach ($queue_items as $item) { $object = $this->objects->getObjectById($item->object_id); - if($object->machine_name === $machine_name && $item->object_level_target === $level) { + if ($object->machine_name === $machine_name && $item->object_level_target === $level) { return true; } } diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index 0c31019b..5bced7fe 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -338,12 +338,12 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane $object = $this->getObjectByMachineName($machine_name); // Check if object prior levels are built or are in build queue - if($level) { + if ($level) { $current_level = $planet->getObjectLevel($object->machine_name); // Check missing levels from build queue - for($i = $current_level + 1; $i < $level; $i++) { - if(!$planet->isBuildingObject($object->machine_name, $i)) { + for ($i = $current_level + 1; $i < $level; $i++) { + if (!$planet->isBuildingObject($object->machine_name, $i)) { return false; } } diff --git a/app/Services/PlanetService.php b/app/Services/PlanetService.php index 33e93619..c6464525 100644 --- a/app/Services/PlanetService.php +++ b/app/Services/PlanetService.php @@ -1469,11 +1469,11 @@ public function isBuildingObject(string $machine_name, int $level): bool $object = $this->objects->getObjectByMachineName($machine_name); // Check only building queue objects - if($object->type !== GameObjectType::Building && $object->type !== GameObjectType::Station) { + if ($object->type !== GameObjectType::Building && $object->type !== GameObjectType::Station) { return false; } - $build_queue = resolve(BuildingQueueService::class); + $build_queue = resolve(BuildingQueueService::class); return $build_queue->objectInBuildingQueue($this, $machine_name, $level); } diff --git a/app/Services/PlayerService.php b/app/Services/PlayerService.php index 776b4505..200b2ef9 100644 --- a/app/Services/PlayerService.php +++ b/app/Services/PlayerService.php @@ -580,7 +580,7 @@ public function delete(): void */ public function isResearchingTech(string $machine_name, int $level): bool { - $research_queue = resolve('OGame\Services\ResearchQueueService'); + $research_queue = resolve('OGame\Services\ResearchQueueService'); return $research_queue->objectInResearchQueue($this, $machine_name, $level); } } diff --git a/app/Services/ResearchQueueService.php b/app/Services/ResearchQueueService.php index 9e6f17da..572ca1e6 100644 --- a/app/Services/ResearchQueueService.php +++ b/app/Services/ResearchQueueService.php @@ -399,7 +399,7 @@ public function cancel(PlayerService $player, int $research_queue_id, int $resea } /** - * Get is object in research queue + * Get is object in research queue * * @return bool */ @@ -410,7 +410,7 @@ public function objectInResearchQueue(PlayerService $player, string $machine_nam foreach ($queue_items as $item) { $object = $this->objects->getObjectById($item->object_id); - if($object->machine_name === $machine_name && $item->object_level_target === $level) { + if ($object->machine_name === $machine_name && $item->object_level_target === $level) { return true; } } diff --git a/tests/Unit/BuildingQueueServiceTest.php b/tests/Unit/BuildingQueueServiceTest.php index 075b97ec..a85ddae2 100644 --- a/tests/Unit/BuildingQueueServiceTest.php +++ b/tests/Unit/BuildingQueueServiceTest.php @@ -38,7 +38,7 @@ protected function setUp(): void } /** - * Tests object is found from building queue + * Tests object is found from building queue */ public function testIsObjectInBuildingQueue(): void { diff --git a/tests/Unit/ResearchQueueServiceTest.php b/tests/Unit/ResearchQueueServiceTest.php index cf2a346b..fd52f4b9 100644 --- a/tests/Unit/ResearchQueueServiceTest.php +++ b/tests/Unit/ResearchQueueServiceTest.php @@ -36,7 +36,7 @@ protected function setUp(): void } /** - * Tests object is found from research queue + * Tests object is found from research queue */ public function testIsObjectInResearchQueue(): void { @@ -79,6 +79,6 @@ public function testCancelItemMissingRequirements(): void $this->research_queue->cancelItemMissingRequirements($this->playerService, $this->planetService); // Assert that impulse drive is in research queue - $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 1)); + $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 1)); } } From 0e19a842489ec1d30552364714a324dac2c85d00 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Sat, 26 Oct 2024 23:25:46 +0300 Subject: [PATCH 03/12] Tests PHPDoc cleanup --- tests/Unit/BuildingQueueServiceTest.php | 2 -- tests/Unit/ObjectServiceTest.php | 2 -- tests/Unit/ResearchQueueServiceTest.php | 2 -- 3 files changed, 6 deletions(-) diff --git a/tests/Unit/BuildingQueueServiceTest.php b/tests/Unit/BuildingQueueServiceTest.php index a85ddae2..5a3b5fcb 100644 --- a/tests/Unit/BuildingQueueServiceTest.php +++ b/tests/Unit/BuildingQueueServiceTest.php @@ -20,8 +20,6 @@ class BuildingQueueServiceTest extends UnitTestCase /** * Set up common test components. - * - * @throws BindingResolutionException */ protected function setUp(): void { diff --git a/tests/Unit/ObjectServiceTest.php b/tests/Unit/ObjectServiceTest.php index fd5e5903..a4f037d5 100644 --- a/tests/Unit/ObjectServiceTest.php +++ b/tests/Unit/ObjectServiceTest.php @@ -18,8 +18,6 @@ class ObjectServiceTest extends UnitTestCase /** * Set up common test components. - * - * @throws BindingResolutionException */ protected function setUp(): void { diff --git a/tests/Unit/ResearchQueueServiceTest.php b/tests/Unit/ResearchQueueServiceTest.php index fd52f4b9..36776a35 100644 --- a/tests/Unit/ResearchQueueServiceTest.php +++ b/tests/Unit/ResearchQueueServiceTest.php @@ -18,8 +18,6 @@ class ResearchQueueServiceTest extends UnitTestCase /** * Set up common test components. - * - * @throws BindingResolutionException */ protected function setUp(): void { From f70c74e6be5fce1a73e9c5c038f42dd75816803f Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Tue, 29 Oct 2024 17:41:05 +0200 Subject: [PATCH 04/12] Fix checking prior research levels as object requirements --- app/Services/ObjectService.php | 3 +- app/Services/ResearchQueueService.php | 46 +++++++++++++-------------- 2 files changed, 24 insertions(+), 25 deletions(-) diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index 5bced7fe..f5013406 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -340,10 +340,9 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane // Check if object prior levels are built or are in build queue if ($level) { $current_level = $planet->getObjectLevel($object->machine_name); - // Check missing levels from build queue for ($i = $current_level + 1; $i < $level; $i++) { - if (!$planet->isBuildingObject($object->machine_name, $i)) { + if (!$planet->isBuildingObject($object->machine_name, $i) && !$player->isResearchingTech($object->machine_name, $i)) { return false; } } diff --git a/app/Services/ResearchQueueService.php b/app/Services/ResearchQueueService.php index 572ca1e6..16367ad7 100644 --- a/app/Services/ResearchQueueService.php +++ b/app/Services/ResearchQueueService.php @@ -105,27 +105,6 @@ public function retrieveFinishedForUser(PlayerService $player): \Illuminate\Supp ->get(); } - /** - * Get research queue items - * - * @return Collection - */ - public function retrieveQueueItems(PlayerService $player): Collection - { - // Fetch queue items from model - return $this->model - ->join('planets', 'research_queues.planet_id', '=', 'planets.id') - ->join('users', 'planets.user_id', '=', 'users.id') - ->where([ - ['users.id', $player->getId()], - ['research_queues.processed', 0], - ['research_queues.canceled', 0], - ]) - ->select('research_queues.*') - ->orderBy('research_queues.time_start', 'asc') - ->get(); - } - /** * Add a research object to the research queue for the current planet. * @@ -405,7 +384,17 @@ public function cancel(PlayerService $player, int $research_queue_id, int $resea */ public function objectInResearchQueue(PlayerService $player, string $machine_name, int $level): bool { - $queue_items = $this->retrieveQueueItems($player); + // Fetch queue items from model + $queue_items = $this->model + ->join('planets', 'research_queues.planet_id', '=', 'planets.id') + ->join('users', 'planets.user_id', '=', 'users.id') + ->where([ + ['users.id', $player->getId()], + ['research_queues.canceled', 0], + ]) + ->select('research_queues.*') + ->orderBy('research_queues.time_start', 'asc') + ->get(); foreach ($queue_items as $item) { $object = $this->objects->getObjectById($item->object_id); @@ -426,7 +415,18 @@ public function objectInResearchQueue(PlayerService $player, string $machine_nam */ public function cancelItemMissingRequirements(PlayerService $player, PlanetService $planet): void { - $research_queue_items = $this->retrieveQueueItems($player); + // Fetch queue items from model + $research_queue_items = $this->model + ->join('planets', 'research_queues.planet_id', '=', 'planets.id') + ->join('users', 'planets.user_id', '=', 'users.id') + ->where([ + ['users.id', $player->getId()], + ['research_queues.processed', 0], + ['research_queues.canceled', 0], + ]) + ->select('research_queues.*') + ->orderBy('research_queues.time_start', 'asc') + ->get(); foreach ($research_queue_items as $research_queue_item) { $object = $this->objects->getObjectById($research_queue_item->object_id); From 4ba6d6024a958c7574dee6ff752569ebfcb10086 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Tue, 29 Oct 2024 19:19:46 +0200 Subject: [PATCH 05/12] Fix getting current level for research --- app/Services/ObjectService.php | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index f5013406..9997e7b3 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -339,7 +339,15 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane // Check if object prior levels are built or are in build queue if ($level) { - $current_level = $planet->getObjectLevel($object->machine_name); + $current_level = 0; + + if($object->type === GameObjectType::Research) { + $current_level = $planet->getPlayer()->getResearchLevel($machine_name); + } + else { + $current_level = $planet->getObjectLevel($object->machine_name); + } + // Check missing levels from build queue for ($i = $current_level + 1; $i < $level; $i++) { if (!$planet->isBuildingObject($object->machine_name, $i) && !$player->isResearchingTech($object->machine_name, $i)) { From fdb1f03410ede3027398440e3354cdd8a9993e78 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Wed, 30 Oct 2024 19:34:08 +0200 Subject: [PATCH 06/12] Extend tests from AccountTestCase --- tests/Unit/BuildingQueueServiceTest.php | 23 +++---------- tests/Unit/ObjectServiceTest.php | 43 ++++++++----------------- tests/Unit/PlanetServiceTest.php | 3 -- tests/Unit/ResearchQueueServiceTest.php | 29 ++++++----------- 4 files changed, 26 insertions(+), 72 deletions(-) diff --git a/tests/Unit/BuildingQueueServiceTest.php b/tests/Unit/BuildingQueueServiceTest.php index 5a3b5fcb..056b869c 100644 --- a/tests/Unit/BuildingQueueServiceTest.php +++ b/tests/Unit/BuildingQueueServiceTest.php @@ -2,37 +2,25 @@ namespace Tests\Unit; -use Illuminate\Foundation\Testing\DatabaseTransactions; use OGame\Models\BuildingQueue; -use OGame\Models\Planet; -use OGame\Models\User; use OGame\Services\BuildingQueueService; use OGame\Services\ObjectService; -use Tests\UnitTestCase; +use Tests\AccountTestCase; -class BuildingQueueServiceTest extends UnitTestCase +class BuildingQueueServiceTest extends AccountTestCase { - use DatabaseTransactions; - protected BuildingQueueService $building_queue; - protected User $player; - /** * Set up common test components. */ + protected function setUp(): void { parent::setUp(); $object_service = new ObjectService(); $this->building_queue = new BuildingQueueService($object_service); - - $planet = Planet::factory()->make(['id' => 1]); - $this->planetService->setPlanet($planet); - - $this->player = User::factory()->make(['id' => 1]); - $this->playerService->load(1); } /** @@ -40,9 +28,6 @@ protected function setUp(): void */ public function testIsObjectInBuildingQueue(): void { - $planet = Planet::factory()->make(['id' => 1]); - $this->planetService->setPlanet($planet); - // Add level 3 shipyard to building queue $queue = new BuildingQueue(); $queue->planet_id = $this->planetService->getPlanetId(); @@ -58,7 +43,7 @@ public function testIsObjectInBuildingQueue(): void /** * Tests building queue item is cancelled if requirements are not met. */ - public function testCancelItemMissingRequirements(): void + public function testCancelObjectMissingRequirements(): void { // Add level 2 robot factory to building queue $queue_robot_factory = new BuildingQueue(); diff --git a/tests/Unit/ObjectServiceTest.php b/tests/Unit/ObjectServiceTest.php index a4f037d5..03d61261 100644 --- a/tests/Unit/ObjectServiceTest.php +++ b/tests/Unit/ObjectServiceTest.php @@ -2,31 +2,23 @@ namespace Tests\Unit; -use Illuminate\Foundation\Testing\DatabaseTransactions; use OGame\Models\BuildingQueue; -use OGame\Models\Planet; +use OGame\Models\Resources; use OGame\Services\ObjectService; -use Tests\UnitTestCase; +use Tests\AccountTestCase; -class ObjectServiceTest extends UnitTestCase +class ObjectServiceTest extends AccountTestCase { - use DatabaseTransactions; - protected ObjectService $object_service; - protected Planet $planet; - /** * Set up common test components. */ - protected function setUp(): void + protected function setUp(): void { parent::setUp(); $this->object_service = new ObjectService(); - - $this->planet = Planet::factory()->make(['id' => 1]); - $this->planetService->setPlanet($this->planet); } /** @@ -34,8 +26,6 @@ protected function setUp(): void */ public function testGetObjectMaxBuildAmount(): void { - $this->createAndSetPlanetModel([]); - // Test with requirements not met $max_build_amount = $this->object_service->getObjectMaxBuildAmount('plasma_turret', $this->planetService, false); $this->assertEquals(0, $max_build_amount); @@ -44,20 +34,13 @@ public function testGetObjectMaxBuildAmount(): void $max_build_amount = $this->object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); $this->assertEquals(1, $max_build_amount); - $this->createAndSetPlanetModel([ - 'small_shield_dome' => 1, - ]); - // Test with object limited to one instance which already exists + $this->planetSetObjectLevel('small_shield_dome', 1); $max_build_amount = $this->object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); $this->assertEquals(0, $max_build_amount); - $this->createAndSetPlanetModel([ - 'metal' => 24000, - 'crystal' => 6000 - ]); - // Test it calculates max amount correctly + $this->planetAddResources(new Resources(24000, 6000, 0, 0)); $max_build_amount = $this->object_service->getObjectMaxBuildAmount('anti_ballistic_missile', $this->planetService, true); $this->assertEquals(3, $max_build_amount); } @@ -69,17 +52,17 @@ public function testGetObjectMaxBuildAmount(): void public function testObjectRequirementsMet(): void { // Assert that requirements are not met if prior levels doesn't exist - $this->assertFalse($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->playerService, 2)); + $this->assertFalse($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->planetService->getPlayer(), 2)); - $this->planet->robot_factory = 1; + $this->planetSetObjectLevel('robot_factory', 1); // Assert that requirements are met if prior levels exists - $this->assertTrue($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->playerService, 2)); + $this->assertTrue($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->planetService->getPlayer(), 2)); // Assert that requirements are not met if requisites are missing - $this->assertFalse($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->playerService, 1)); + $this->assertFalse($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->planetService->getPlayer(), 1)); - // Add shipyard to build queue + // Add to build queue $queue = new BuildingQueue(); $queue->planet_id = $this->planetService->getPlanetId(); $queue->object_id = 21; @@ -87,9 +70,9 @@ public function testObjectRequirementsMet(): void $queue->save(); // Assert that requirements are met if requisites are in build queue - $this->assertTrue($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->playerService, 1)); + $this->assertTrue($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->planetService->getPlayer(), 1)); // Assert that research requirements are not met if building requirements are not met - $this->assertFalse($this->object_service->objectRequirementsMet('computer_technology', $this->planetService, $this->playerService, 1)); + $this->assertFalse($this->object_service->objectRequirementsMet('computer_technology', $this->planetService, $this->planetService->getPlayer(), 1)); } } diff --git a/tests/Unit/PlanetServiceTest.php b/tests/Unit/PlanetServiceTest.php index 138c0aab..c58515e9 100644 --- a/tests/Unit/PlanetServiceTest.php +++ b/tests/Unit/PlanetServiceTest.php @@ -3,7 +3,6 @@ namespace Tests\Unit; use Illuminate\Contracts\Container\BindingResolutionException; -use Illuminate\Foundation\Testing\DatabaseTransactions; use OGame\Models\BuildingQueue; use OGame\Models\Enums\ResourceType; use OGame\Models\Planet; @@ -12,8 +11,6 @@ class PlanetServiceTest extends UnitTestCase { - use DatabaseTransactions; - /** * Set up common test components. * diff --git a/tests/Unit/ResearchQueueServiceTest.php b/tests/Unit/ResearchQueueServiceTest.php index 36776a35..f7888d20 100644 --- a/tests/Unit/ResearchQueueServiceTest.php +++ b/tests/Unit/ResearchQueueServiceTest.php @@ -2,18 +2,13 @@ namespace Tests\Unit; -use Illuminate\Foundation\Testing\DatabaseTransactions; -use OGame\Models\Planet; use OGame\Models\ResearchQueue; -use OGame\Models\User; use OGame\Services\ObjectService; use OGame\Services\ResearchQueueService; -use Tests\UnitTestCase; +use Tests\AccountTestCase; -class ResearchQueueServiceTest extends UnitTestCase +class ResearchQueueServiceTest extends AccountTestCase { - use DatabaseTransactions; - protected ResearchQueueService $research_queue; /** @@ -25,12 +20,6 @@ protected function setUp(): void $object_service = new ObjectService(); $this->research_queue = new ResearchQueueService($object_service); - - $planet = Planet::factory()->make(['id' => 1]); - $this->planetService->setPlanet($planet); - - User::factory()->make(['id' => 1]); - $this->playerService->load(1); } /** @@ -45,9 +34,9 @@ public function testIsObjectInResearchQueue(): void $queue->object_level_target = 3; $queue->save(); - $this->assertTrue($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 3)); - $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 4)); - $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'energy_technology', 4)); + $this->assertTrue($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 3)); + $this->assertFalse($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 4)); + $this->assertFalse($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'energy_technology', 4)); } /** @@ -70,13 +59,13 @@ public function testCancelItemMissingRequirements(): void $queue->save(); // Assert that impulse drive is in research queue - $this->assertTrue($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 1)); + $this->assertTrue($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 1)); // Cancel energy technology - $this->research_queue->cancel($this->playerService, $queue_energy_tech->id, 113); - $this->research_queue->cancelItemMissingRequirements($this->playerService, $this->planetService); + $this->research_queue->cancel($this->planetService->getPlayer(), $queue_energy_tech->id, 113); + $this->research_queue->cancelItemMissingRequirements($this->planetService->getPlayer(), $this->planetService); // Assert that impulse drive is in research queue - $this->assertFalse($this->research_queue->objectInResearchQueue($this->playerService, 'impulse_drive', 1)); + $this->assertFalse($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 1)); } } From ffab787c8cc023121df71c5138ccb0cb44ad7a48 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Wed, 30 Oct 2024 19:45:23 +0200 Subject: [PATCH 07/12] Fix psr styling issues --- app/Services/ObjectService.php | 5 ++--- tests/Unit/BuildingQueueServiceTest.php | 1 - tests/Unit/ObjectServiceTest.php | 2 +- 3 files changed, 3 insertions(+), 5 deletions(-) diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index 9997e7b3..372fab0f 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -341,10 +341,9 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane if ($level) { $current_level = 0; - if($object->type === GameObjectType::Research) { + if ($object->type === GameObjectType::Research) { $current_level = $planet->getPlayer()->getResearchLevel($machine_name); - } - else { + } else { $current_level = $planet->getObjectLevel($object->machine_name); } diff --git a/tests/Unit/BuildingQueueServiceTest.php b/tests/Unit/BuildingQueueServiceTest.php index 056b869c..fa28bf18 100644 --- a/tests/Unit/BuildingQueueServiceTest.php +++ b/tests/Unit/BuildingQueueServiceTest.php @@ -14,7 +14,6 @@ class BuildingQueueServiceTest extends AccountTestCase /** * Set up common test components. */ - protected function setUp(): void { parent::setUp(); diff --git a/tests/Unit/ObjectServiceTest.php b/tests/Unit/ObjectServiceTest.php index 03d61261..e104b6f1 100644 --- a/tests/Unit/ObjectServiceTest.php +++ b/tests/Unit/ObjectServiceTest.php @@ -14,7 +14,7 @@ class ObjectServiceTest extends AccountTestCase /** * Set up common test components. */ - protected function setUp(): void + protected function setUp(): void { parent::setUp(); From 7e62cfc833dfe77f9883b774240c0eed43d8d014 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Wed, 30 Oct 2024 20:28:49 +0200 Subject: [PATCH 08/12] Fix object requirements check for queue items --- app/Services/BuildingQueueService.php | 2 +- app/Services/ObjectService.php | 9 +++++---- app/Services/ResearchQueueService.php | 2 +- 3 files changed, 7 insertions(+), 6 deletions(-) diff --git a/app/Services/BuildingQueueService.php b/app/Services/BuildingQueueService.php index 81748bf2..6ab5cb98 100644 --- a/app/Services/BuildingQueueService.php +++ b/app/Services/BuildingQueueService.php @@ -243,7 +243,7 @@ public function start(PlanetService $planet, int $time_start = 0): void // Sanity check: check if the building requirements are still met. If not, // then cancel build request. - if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer(), $queue_item->object_level_target)) { + if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer(), $queue_item->object_level_target, false)) { $this->cancel($planet, $queue_item->id, $queue_item->object_id); continue; diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index 372fab0f..aabc1dc0 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -330,9 +330,10 @@ public function getBuildingObjectsWithStorage(): array * @param PlanetService $planet * @param PlayerService $player * @param int $level + * @param bool $queued * @return bool */ - public function objectRequirementsMet(string $machine_name, PlanetService $planet, PlayerService $player, int $level = 0): bool + public function objectRequirementsMet(string $machine_name, PlanetService $planet, PlayerService $player, int $level = 0, bool $queued = true): bool { try { $object = $this->getObjectByMachineName($machine_name); @@ -349,7 +350,7 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane // Check missing levels from build queue for ($i = $current_level + 1; $i < $level; $i++) { - if (!$planet->isBuildingObject($object->machine_name, $i) && !$player->isResearchingTech($object->machine_name, $i)) { + if (!$planet->isBuildingObject($object->machine_name, $i) && (!$queued || !$player->isResearchingTech($object->machine_name, $i))) { return false; } } @@ -359,13 +360,13 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane // Load required object and check if requirements are met. $object_required = $this->getObjectByMachineName($requirement->object_machine_name); if ($object_required->type === GameObjectType::Research) { - if ($player->getResearchLevel($object_required->machine_name) < $requirement->level && !$player->isResearchingTech($requirement->object_machine_name, $requirement->level)) { + if ($player->getResearchLevel($object_required->machine_name) < $requirement->level && (!$queued || !$player->isResearchingTech($requirement->object_machine_name, $requirement->level))) { return false; } } else { // Check if requirements are met with existing buildings or with buildings on build queue. // Building queue is checked only for building queue objects, not for unit queue objects. - if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level && !$planet->isBuildingObject($requirement->object_machine_name, $requirement->level)) { + if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level && (!$queued || !$planet->isBuildingObject($requirement->object_machine_name, $requirement->level))) { return false; } } diff --git a/app/Services/ResearchQueueService.php b/app/Services/ResearchQueueService.php index 16367ad7..4e1348b9 100644 --- a/app/Services/ResearchQueueService.php +++ b/app/Services/ResearchQueueService.php @@ -289,7 +289,7 @@ public function start(PlayerService $player, int $time_start = 0): void // Sanity check: check if the researching requirements are still met. If not, // then cancel research request. - if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $player, $queue_item->object_level_target)) { + if (!$this->objects->objectRequirementsMet($object->machine_name, $planet, $player, $queue_item->object_level_target, false)) { $this->cancel($player, $queue_item->id, $queue_item->object_id); continue; From 1145cfcaa931e576856cb03cae40cc23e7d3d8df Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Thu, 31 Oct 2024 16:26:51 +0200 Subject: [PATCH 09/12] Fix research multi queue test --- tests/Feature/ResearchQueueTest.php | 1 + 1 file changed, 1 insertion(+) diff --git a/tests/Feature/ResearchQueueTest.php b/tests/Feature/ResearchQueueTest.php index 4b6ba031..0fed0a5e 100644 --- a/tests/Feature/ResearchQueueTest.php +++ b/tests/Feature/ResearchQueueTest.php @@ -109,6 +109,7 @@ public function testResearchQueueMultiQueue(): void Carbon::setTestNow($testTime); // Check if one of the research items is finished and is now level 1. + $this->planetService->getPlayer()->updateResearchQueue(); $response = $this->get('/research'); $response->assertStatus(200); $this->assertObjectLevelOnPage($response, 'energy_technology', 1, 'Energy technology is not at level one 15 minutes after build request issued.'); From aecf54f680140f98d1bf0e4b33cfa7adf743e3ee Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Thu, 31 Oct 2024 19:05:44 +0200 Subject: [PATCH 10/12] Prevent starting research without research lab --- app/Services/ObjectService.php | 67 +++++++++++++++++++++------ app/Services/ResearchQueueService.php | 1 - 2 files changed, 54 insertions(+), 14 deletions(-) diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index aabc1dc0..6e748343 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -338,21 +338,19 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane try { $object = $this->getObjectByMachineName($machine_name); - // Check if object prior levels are built or are in build queue - if ($level) { - $current_level = 0; + // Check that planet has research lab for research objects + if ($object->type === GameObjectType::Research) { + $research_lab = $planet->getObjectLevel('research_lab'); - if ($object->type === GameObjectType::Research) { - $current_level = $planet->getPlayer()->getResearchLevel($machine_name); - } else { - $current_level = $planet->getObjectLevel($object->machine_name); + if (!$research_lab) { + return false; } + } - // Check missing levels from build queue - for ($i = $current_level + 1; $i < $level; $i++) { - if (!$planet->isBuildingObject($object->machine_name, $i) && (!$queued || !$player->isResearchingTech($object->machine_name, $i))) { - return false; - } + // Check required prior levels + if ($level) { + if (!$this->objectLevelsMet($object, $planet, $player, $level, $queued)) { + return false; } } @@ -360,11 +358,12 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane // Load required object and check if requirements are met. $object_required = $this->getObjectByMachineName($requirement->object_machine_name); if ($object_required->type === GameObjectType::Research) { + // Check if requirements are met with existing technology or with research items in build queue. if ($player->getResearchLevel($object_required->machine_name) < $requirement->level && (!$queued || !$player->isResearchingTech($requirement->object_machine_name, $requirement->level))) { return false; } } else { - // Check if requirements are met with existing buildings or with buildings on build queue. + // Check if requirements are met with existing buildings or with buildings in build queue. // Building queue is checked only for building queue objects, not for unit queue objects. if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level && (!$queued || !$planet->isBuildingObject($requirement->object_machine_name, $requirement->level))) { return false; @@ -509,4 +508,46 @@ public function getObjectRawPrice(string $machine_name, int $level = 0): Resourc return new Resources($metal, $crystal, $deuterium, $energy); } + + /** + * Check if object prior level requirements are met (for building it). + * Prior levels can be already built or in queues + * + * @param GameObject $object + * @param PlanetService $planet + * @param PlayerService $player + * @param int $level + * @param bool $queued + * @return bool + */ + private function objectLevelsMet(GameObject $object, PlanetService $planet, PlayerService $player, int $level, bool $queued): bool + { + $current_level = 0; + + if ($object->type === GameObjectType::Research) { + $current_level = $planet->getPlayer()->getResearchLevel($object->machine_name); + } else { + $current_level = $planet->getObjectLevel($object->machine_name); + } + + // Check if target level is next level + if ($current_level + 1 === $level) { + return true; + } + + // Check if items in queues should be included or not + if (!$queued) { + // There are prior levels, but queue should not be included + return false; + } + + // Check prior levels from queues + for ($i = $current_level + 1; $i < $level; $i++) { + if (!$planet->isBuildingObject($object->machine_name, $i) && !$player->isResearchingTech($object->machine_name, $i)) { + return false; + } + } + + return true; + } } diff --git a/app/Services/ResearchQueueService.php b/app/Services/ResearchQueueService.php index 4e1348b9..5592810d 100644 --- a/app/Services/ResearchQueueService.php +++ b/app/Services/ResearchQueueService.php @@ -421,7 +421,6 @@ public function cancelItemMissingRequirements(PlayerService $player, PlanetServi ->join('users', 'planets.user_id', '=', 'users.id') ->where([ ['users.id', $player->getId()], - ['research_queues.processed', 0], ['research_queues.canceled', 0], ]) ->select('research_queues.*') From b72f6dda05868c0f7365ef8e9c5278a6f2dce7eb Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Thu, 31 Oct 2024 19:42:17 +0200 Subject: [PATCH 11/12] Require research lab level to be ready for tech --- app/Services/ObjectService.php | 20 +++++++++----------- 1 file changed, 9 insertions(+), 11 deletions(-) diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index 6e748343..bb8aca9f 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -338,15 +338,6 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane try { $object = $this->getObjectByMachineName($machine_name); - // Check that planet has research lab for research objects - if ($object->type === GameObjectType::Research) { - $research_lab = $planet->getObjectLevel('research_lab'); - - if (!$research_lab) { - return false; - } - } - // Check required prior levels if ($level) { if (!$this->objectLevelsMet($object, $planet, $player, $level, $queued)) { @@ -357,15 +348,22 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane foreach ($object->requirements as $requirement) { // Load required object and check if requirements are met. $object_required = $this->getObjectByMachineName($requirement->object_machine_name); + $check_queue = $queued; + + // SKip queue check for research lab as it must be present for research objects + if ($object_required->machine_name === 'research_lab') { + $check_queue = false; + } + if ($object_required->type === GameObjectType::Research) { // Check if requirements are met with existing technology or with research items in build queue. - if ($player->getResearchLevel($object_required->machine_name) < $requirement->level && (!$queued || !$player->isResearchingTech($requirement->object_machine_name, $requirement->level))) { + if ($player->getResearchLevel($object_required->machine_name) < $requirement->level && (!$check_queue || !$player->isResearchingTech($requirement->object_machine_name, $requirement->level))) { return false; } } else { // Check if requirements are met with existing buildings or with buildings in build queue. // Building queue is checked only for building queue objects, not for unit queue objects. - if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level && (!$queued || !$planet->isBuildingObject($requirement->object_machine_name, $requirement->level))) { + if ($planet->getObjectLevel($object_required->machine_name) < $requirement->level && (!$check_queue || !$planet->isBuildingObject($requirement->object_machine_name, $requirement->level))) { return false; } } From 0064cfef83d30f6908b3375bd85553b35d20eae0 Mon Sep 17 00:00:00 2001 From: Mika Rautanen Date: Sun, 3 Nov 2024 17:11:36 +0200 Subject: [PATCH 12/12] Fix unit queue requirements check and update tests --- .../Abstracts/AbstractUnitsController.php | 2 +- app/Services/BuildingQueueService.php | 4 +- app/Services/ObjectService.php | 2 +- app/Services/ResearchQueueService.php | 20 +++- app/Services/UnitQueueService.php | 2 +- tests/AccountTestCase.php | 68 ++++++++++- tests/Feature/BuildQueueCancelTest.php | 53 ++++++++- tests/Feature/BuildQueueTest.php | 34 ++++++ tests/Feature/ResearchQueueCancelTest.php | 53 ++++++++- tests/Feature/ResearchQueueTest.php | 43 ++++++- tests/Feature/UnitQueueTest.php | 109 ++++++++++++++++++ tests/Unit/BuildingQueueServiceTest.php | 71 ------------ tests/Unit/ObjectServiceTest.php | 73 +++--------- tests/Unit/PlanetServiceTest.php | 8 +- tests/Unit/ResearchQueueServiceTest.php | 71 ------------ 15 files changed, 392 insertions(+), 221 deletions(-) delete mode 100644 tests/Unit/BuildingQueueServiceTest.php delete mode 100644 tests/Unit/ResearchQueueServiceTest.php diff --git a/app/Http/Controllers/Abstracts/AbstractUnitsController.php b/app/Http/Controllers/Abstracts/AbstractUnitsController.php index 520b6f0d..e1909137 100644 --- a/app/Http/Controllers/Abstracts/AbstractUnitsController.php +++ b/app/Http/Controllers/Abstracts/AbstractUnitsController.php @@ -94,7 +94,7 @@ public function index(Request $request, PlayerService $player, ObjectService $ob $amount = $planet->getObjectAmount($object->machine_name); // Check requirements of this building - $requirements_met = $objects->objectRequirementsMet($object->machine_name, $planet, $player); + $requirements_met = $objects->objectRequirementsMet($object->machine_name, $planet, $player, 0, false); // Check if the current planet has enough resources to build this building. $enough_resources = $planet->hasResources($objects->getObjectPrice($object->machine_name, $planet)); diff --git a/app/Services/BuildingQueueService.php b/app/Services/BuildingQueueService.php index 6ab5cb98..68297853 100644 --- a/app/Services/BuildingQueueService.php +++ b/app/Services/BuildingQueueService.php @@ -84,8 +84,6 @@ public function add(PlanetService $planet, int $building_id): void { $build_queue = $this->retrieveQueue($planet); - $building = $this->objects->getObjectById($building_id); - // Max amount of buildings that can be in the queue in a given time. // TODO: refactor throw exception into a more user-friendly message. if ($build_queue->isQueueFull()) { @@ -93,6 +91,8 @@ public function add(PlanetService $planet, int $building_id): void throw new Exception('Maximum number of items already in queue.'); } + $building = $this->objects->getObjectById($building_id); + // Check if user satisifes requirements to build this object. // TODO: refactor throw exception into a more user-friendly message. $requirements_met = $this->objects->objectRequirementsMet($building->machine_name, $planet, $planet->getPlayer()); diff --git a/app/Services/ObjectService.php b/app/Services/ObjectService.php index bb8aca9f..0ba6272b 100644 --- a/app/Services/ObjectService.php +++ b/app/Services/ObjectService.php @@ -350,7 +350,7 @@ public function objectRequirementsMet(string $machine_name, PlanetService $plane $object_required = $this->getObjectByMachineName($requirement->object_machine_name); $check_queue = $queued; - // SKip queue check for research lab as it must be present for research objects + // Skip queue check for research lab as it must be present for research objects if ($object_required->machine_name === 'research_lab') { $check_queue = false; } diff --git a/app/Services/ResearchQueueService.php b/app/Services/ResearchQueueService.php index 5592810d..2261676b 100644 --- a/app/Services/ResearchQueueService.php +++ b/app/Services/ResearchQueueService.php @@ -116,21 +116,29 @@ public function retrieveFinishedForUser(PlayerService $player): \Illuminate\Supp */ public function add(PlayerService $player, PlanetService $planet, int $research_object_id): void { - $build_queue = $this->retrieveQueue($planet); + $research_queue = $this->retrieveQueue($planet); - // Max amount of buildings that can be in the queue at a given time. - if ($build_queue->isQueueFull()) { - // Max amount of build queue items already exist, throw exception. + // Max amount of research items that can be in the queue at a given time. + // TODO: refactor throw exception into a more user-friendly message. + if ($research_queue->isQueueFull()) { + // Max amount of research queue items already exist, throw exception. throw new Exception('Maximum number of items already in queue.'); } $object = $this->objects->getResearchObjectById($research_object_id); + // Check if user satisifes requirements to research this object. + // TODO: refactor throw exception into a more user-friendly message. + $requirements_met = $this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer()); + if (!$requirements_met) { + throw new Exception('Requirements not met to build this object.'); + } + // @TODO: add checks that current logged in user is owner of planet - // and is able to add this object to the building queue. + // and is able to add this object to the research queue. $current_level = $player->getResearchLevel($object->machine_name); - // Check to see how many other items of this building there are already + // Check to see how many other items of this technology there are already // in the queue, because if so then the level needs to be higher than that. $amount = $this->activeBuildingQueueItemCount($player, $research_object_id); $next_level = $current_level + $amount + 1; diff --git a/app/Services/UnitQueueService.php b/app/Services/UnitQueueService.php index 62918e93..03c6d77d 100644 --- a/app/Services/UnitQueueService.php +++ b/app/Services/UnitQueueService.php @@ -159,7 +159,7 @@ public function add(PlanetService $planet, int $object_id, int $requested_build_ $object = $this->objects->getUnitObjectById($object_id); // Check if user satisifes requirements to build this object. - $requirements_met = $this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer()); + $requirements_met = $this->objects->objectRequirementsMet($object->machine_name, $planet, $planet->getPlayer(), 0, false); // Sanity check: check if the planet has enough resources to build // the amount requested. If not, then adjust the ordered amount. diff --git a/tests/AccountTestCase.php b/tests/AccountTestCase.php index dbc63614..5ecb9ba2 100644 --- a/tests/AccountTestCase.php +++ b/tests/AccountTestCase.php @@ -384,7 +384,7 @@ protected function assertResourcesOnPage(TestResponse $response, Resources $reso } } - protected function assertObjectInQueue(TestResponse $response, string $machine_name, string $error_message = ''): void + protected function assertObjectInQueue(TestResponse $response, string $machine_name, int $level, string $error_message = ''): void { // Get object name from machine name. try { @@ -399,7 +399,7 @@ protected function assertObjectInQueue(TestResponse $response, string $machine_n if (!$responseContent) { $responseContent = ''; } - $condition1 = str_contains($responseContent, 'Cancel production of ' . $object->title); + $condition1 = str_contains($responseContent, 'Cancel production of ' . $object->title . ' level '. $level); $condition2 = str_contains($responseContent, 'do you really want to cancel ' . $object->title); $this->assertTrue($condition1 || $condition2, 'Neither of the expected texts were found in the response.'); } catch (Exception $e) { @@ -432,6 +432,70 @@ protected function assertObjectNotInQueue(TestResponse $response, string $machin } } + protected function assertEmptyBuildingQueue(TestResponse $response, string $error_message = ''): void + { + // Check if "no buildings being built" text is present on page. + try { + $responseContent = $response->getContent(); + if (!$responseContent) { + $responseContent = ''; + } + $condition = str_contains($responseContent, 'no building being built'); + $this->assertTrue($condition, 'expected text was not found in the response.'); + } catch (Exception $e) { + if (!empty($error_message)) { + $this->fail($error_message . '. Error: ' . $e->getMessage()); + } else { + $this->fail('Building queue is not empty. Error: ' . $e->getMessage()); + } + } + } + + protected function assertEmptyResearchQueue(TestResponse $response, string $error_message = ''): void + { + // Check if "no research done" text is present on page. + try { + $responseContent = $response->getContent(); + if (!$responseContent) { + $responseContent = ''; + } + $condition = str_contains($responseContent, 'no research done'); + $this->assertTrue($condition, 'expected text was not found in the response.'); + } catch (Exception $e) { + if (!empty($error_message)) { + $this->fail($error_message . '. Error: ' . $e->getMessage()); + } else { + $this->fail('Research queue is not empty. Error: ' . $e->getMessage()); + } + } + } + + protected function assertRequirementsNotMet(TestResponse $response, string $machine_name, string $error_message = ''): void + { + // Get object name from machine name. + try { + $object = $this->planetService->objects->getObjectByMachineName($machine_name); + } catch (Exception $e) { + $this->fail('Failed to get object by machine name: ' . $machine_name . '. Error: ' . $e->getMessage()); + } + + // Check if "Requirements are not met" text is present on page. + try { + $responseContent = $response->getContent(); + if (!$responseContent) { + $responseContent = ''; + } + $condition = str_contains($responseContent, $object->title.'
Requirements are not met!'); + $this->assertTrue($condition, 'expected text was not found in the response.'); + } catch (Exception $e) { + if (!empty($error_message)) { + $this->fail($error_message . '. Error: ' . $e->getMessage()); + } else { + $this->fail('Requirements are met. Error: ' . $e->getMessage()); + } + } + } + /** * Add a resource build request to the current users current planet. * @param string $machine_name diff --git a/tests/Feature/BuildQueueCancelTest.php b/tests/Feature/BuildQueueCancelTest.php index b23f0be3..0fcb9267 100644 --- a/tests/Feature/BuildQueueCancelTest.php +++ b/tests/Feature/BuildQueueCancelTest.php @@ -35,7 +35,7 @@ public function testBuildQueueCancelMultiple(): void Carbon::setTestNow($testTime); $response = $this->get('/resources'); - $this->assertObjectInQueue($response, 'metal_mine', 'Metal mine is expected in build queue but cannot be found.'); + $this->assertObjectInQueue($response, 'metal_mine', 3, 'Metal mine level 3 is expected in build queue but cannot be found.'); // Extract first and second number on page which looks like this where num1/num2 are ints: // "cancelProduction(num1,num2," @@ -95,7 +95,7 @@ public function testBuildQueueCancelRefundResources(): void $this->addResourceBuildRequest('metal_mine'); $response = $this->get('/resources'); - $this->assertObjectInQueue($response, 'metal_mine', 'Metal mine is not in build queue.'); + $this->assertObjectInQueue($response, 'metal_mine', 1, 'Metal mine level 1 is not in build queue.'); // Extract first and second number on page which looks like this where num1/num2 are ints: // "cancelProduction(num1,num2," @@ -149,7 +149,7 @@ public function testBuildQueueCancelSecondEntry(): void // Extract first and second number on page which looks like this where num1/num2 are ints: // "cancelbuilding(num1,num2," - $this->assertObjectInQueue($response, 'crystal_mine', 'Crystal mine is not in build queue.'); + $this->assertObjectInQueue($response, 'crystal_mine', 1, 'Crystal mine level 1 is not in build queue.'); // Extract the content from the response $pageContent = $response->getContent(); @@ -178,4 +178,51 @@ public function testBuildQueueCancelSecondEntry(): void $this->throwException(new BindingResolutionException('Less than two "cancelProduction" calls found.')); } } + + /** + * Tests building queue item is cancelled if requirements are not met. + */ + public function testCancelObjectMissingRequirements(): void + { + // Assert that build queue is empty + $response = $this->get('/facilities'); + $response->assertStatus(200); + + $this->assertEmptyBuildingQueue($response); + + // Add resource to build required facilities to planet + $this->planetAddResources(new Resources(5000, 5000, 5000, 0)); + + // Add required facilities to building queue + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('shipyard'); + + $response = $this->get('/facilities'); + $this->assertObjectInQueue($response, 'shipyard', 1, 'Shipyard level 1 is not in build queue.'); + + // Extract the first and second number from the first cancelbuilding call + $cancelProductionCall = $response->getContent(); + if (empty($cancelProductionCall)) { + $cancelProductionCall = ''; + } + $cancelProductionCall = explode('onclick="cancelbuilding(', $cancelProductionCall); + $cancelProductionCall = explode(',', $cancelProductionCall[1]); + $number1 = (int)$cancelProductionCall[0]; + $number2 = (int)$cancelProductionCall[1]; + + // Check if both numbers are integers. If not, throw an exception. + if (empty($number1) || empty($number2)) { + throw new BindingResolutionException('Could not extract the building queue ID from the page.'); + } + + // Cancel Robotics Factory level 1, this will cancel also Robotics Factory level 2 and Shipyard level 1 + $this->cancelFacilitiesBuildRequest($number1, $number2); + + // Assert that building queue is empty + $response = $this->get('/facilities'); + $response->assertStatus(200); + + $this->assertEmptyBuildingQueue($response); + } } diff --git a/tests/Feature/BuildQueueTest.php b/tests/Feature/BuildQueueTest.php index 1305a7f2..2d59b788 100644 --- a/tests/Feature/BuildQueueTest.php +++ b/tests/Feature/BuildQueueTest.php @@ -221,6 +221,40 @@ public function testBuildQueueFailUnfulfilledRequirements(): void $this->assertObjectLevelOnPage($response, 'fusion_plant', 0, 'Fusion Reactor has been built while player has not satisfied building requirements.'); } + /** + * Verify that shipyard can be queued when robotics factory is in queue. + * @throws Exception + */ + public function testBuildQueueFacilitiesShipyardQueuedRequirements(): void + { + // Set the current time to a specific moment for testing + $testTime = Carbon::create(2024, 1, 1, 12, 0, 0); + Carbon::setTestNow($testTime); + + // Add resource to build required facilities to planet + $this->planetAddResources(new Resources(5000, 5000, 5000, 0)); + + // Assert that building requirements for Shipyard are not met as Robotics Factory is missing + $response = $this->get('/facilities'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'shipyard', 'Shipyard building requirements not met.'); + + // Add Robotics Factory level 1 and 2 to build queue + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('robot_factory'); + + // Add Shipyard level 1 to queue + $this->addFacilitiesBuildRequest('shipyard'); + + // Verify the research is finished 10 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 10, 0); + Carbon::setTestNow($testTime); + + $response = $this->get('/facilities'); + $response->assertStatus(200); + $this->assertObjectLevelOnPage($response, 'shipyard', 1, 'Shipyard is not at level one 10 minutes after build request issued.'); + } + /** * Verify that building construction time is calculated correctly (higher than 0) * @throws Exception diff --git a/tests/Feature/ResearchQueueCancelTest.php b/tests/Feature/ResearchQueueCancelTest.php index 6cb5875c..04da5b1b 100644 --- a/tests/Feature/ResearchQueueCancelTest.php +++ b/tests/Feature/ResearchQueueCancelTest.php @@ -3,6 +3,7 @@ namespace Tests\Feature; use Exception; +use Illuminate\Contracts\Container\BindingResolutionException; use Illuminate\Support\Carbon; use Illuminate\Testing\TestResponse; use OGame\Models\Resources; @@ -39,7 +40,7 @@ public function testResearchQueueCancelMultiple(): void Carbon::setTestNow($testTime); $response = $this->get('/research'); - $this->assertObjectInQueue($response, 'energy_technology', 'Energy Technology is expected in build queue but cannot be found.'); + $this->assertObjectInQueue($response, 'energy_technology', 3, 'Energy Technology level 3 is expected in build queue but cannot be found.'); $this->pressCancelButtonOnPage($response); @@ -81,7 +82,7 @@ public function testResearchQueueCancelRefundResources(): void $response = $this->get('/research'); $response->assertStatus(200); - $this->assertObjectInQueue($response, 'energy_technology', 'Energy Technology is not in build queue.'); + $this->assertObjectInQueue($response, 'energy_technology', 1, 'Energy Technology level 1 is not in build queue.'); $this->pressCancelButtonOnPage($response); @@ -116,7 +117,7 @@ public function testBuildQueueCancelSecondEntry(): void $response = $this->get('/research'); $response->assertStatus(200); - $this->assertObjectInQueue($response, 'computer_technology', 'Computer Technology is not in build queue.'); + $this->assertObjectInQueue($response, 'computer_technology', 1, 'Computer Technology level 1 is not in build queue.'); // Extract the content from the response $pageContent = $response->getContent(); @@ -208,4 +209,50 @@ private function pressCancelButtonOnPage(TestResponse $response): void // Do POST to cancel build queue item: $this->cancelResearchBuildRequest($number1, $number2); } + + /** + * Tests research queue item is cancelled if requirements are not met. + */ + public function testCancelResearchMissingRequirements(): void + { + // Assert that research queue is empty + $response = $this->get('/research'); + $response->assertStatus(200); + + $this->assertEmptyResearchQueue($response); + + // Set facilities and add resources to planet that test requires. + $this->planetSetObjectLevel('research_lab', 2); + $this->planetAddResources(new Resources(5000, 5000, 5000, 0)); + + $this->addResearchBuildRequest('energy_technology'); + $this->addResearchBuildRequest('impulse_drive'); + + $response = $this->get('/research'); + $this->assertObjectInQueue($response, 'impulse_drive', 1, 'Impulse Drive 1 is not in research queue.'); + + // Extract the first and second number from the first cancelbuilding call + $cancelProductionCall = $response->getContent(); + if (empty($cancelProductionCall)) { + $cancelProductionCall = ''; + } + $cancelProductionCall = explode('onclick="cancelbuilding(', $cancelProductionCall); + $cancelProductionCall = explode(',', $cancelProductionCall[1]); + $number1 = (int)$cancelProductionCall[0]; + $number2 = (int)$cancelProductionCall[1]; + + // Check if both numbers are integers. If not, throw an exception. + if (empty($number1) || empty($number2)) { + throw new BindingResolutionException('Could not extract the building queue ID from the page.'); + } + + // Cancel Energy technology level 1, this will cancel also Impulse Drive level 1 + $this->cancelResearchBuildRequest($number1, $number2); + + // Assert that building queue is empty + $response = $this->get('/research'); + $response->assertStatus(200); + + $this->assertEmptyResearchQueue($response); + } } diff --git a/tests/Feature/ResearchQueueTest.php b/tests/Feature/ResearchQueueTest.php index 0fed0a5e..61a866fb 100644 --- a/tests/Feature/ResearchQueueTest.php +++ b/tests/Feature/ResearchQueueTest.php @@ -155,7 +155,7 @@ public function testResearchQueueCancelRefundResources(): void // Assert that resources have been actually deducted $this->assertResourcesOnPage($response, new Resources(0, 500, 0, 0)); // Assert that the research is in the queue - $this->assertObjectInQueue($response, 'energy_technology', 'Energy Technology is not in build queue.'); + $this->assertObjectInQueue($response, 'energy_technology', 1, 'Energy Technology level 1 is not in build queue.'); // Extract first and second number on page which looks like this where num1/num2 are ints: // "cancelProduction(num1,num2," @@ -198,4 +198,45 @@ public function testResearchProductionTime(): void $research_time = $this->planetService->getTechnologyResearchTime('energy_technology'); $this->assertGreaterThan(0, $research_time); } + + /** + * Verify that research lab requirement is working for research objects. + * @throws Exception + */ + public function testResearchLabRequirement(): void + { + // Set the current time to a specific moment for testing + $testTime = Carbon::create(2024, 1, 1, 12, 0, 0); + Carbon::setTestNow($testTime); + + // Add required resources for research to planet + $this->planetAddResources(new Resources(5000, 5000, 5000, 0)); + + // Assert that research requirements for Energy Technology are not met as Research Lab is missing + $response = $this->get('/research'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'energy_technology', 'Energy Technology research requirements not met.'); + + // Add Research Lab level 1 to build queue + $this->addFacilitiesBuildRequest('research_lab'); + + // Assert that research requirements for Energy Technology are not met as Research Lab is in build queue + $response = $this->get('/research'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'energy_technology', 'Energy Technology research requirements not met.'); + + // Verify that Energy Technology can be added to research queue 2 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 2, 0); + Carbon::setTestNow($testTime); + + $this->addResearchBuildRequest('energy_technology'); + + // Verify the research is finished 2 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 4, 0); + Carbon::setTestNow($testTime); + + $response = $this->get('/research'); + $response->assertStatus(200); + $this->assertObjectLevelOnPage($response, 'energy_technology', 1, 'Energy technology is not at level one 2 minutes after build request issued.'); + } } diff --git a/tests/Feature/UnitQueueTest.php b/tests/Feature/UnitQueueTest.php index dadd5bf4..bdb07bab 100644 --- a/tests/Feature/UnitQueueTest.php +++ b/tests/Feature/UnitQueueTest.php @@ -374,6 +374,115 @@ public function testUnitQueueInsufficientResources(): void $this->assertResourcesOnPage($response, new Resources(15500, 5500, 0, 0)); } + /** + * Verify that shipyard requirement is working for unit objects. + * @throws Exception + */ + public function testUnitQueueShipyardRequirement(): void + { + // Set the current time to a specific moment for testing + $testTime = Carbon::create(2024, 1, 1, 12, 0, 0); + Carbon::setTestNow($testTime); + + // Add required resources to planet + $this->planetAddResources(new Resources(5000, 5000, 5000, 0)); + + // Assert that build requirements for Solar Satellite are not met as Shipyard is missing + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'solar_satellite', 'Solar Satellite build requirements are met.'); + + // Add Shipyard level 1 with requisities to build queue + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('shipyard'); + + // Assert that build requirements for Solar Satellite are not met as Shipyard is in build queue + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'solar_satellite', 'Solar Satellite build requirements are met.'); + + // Verify that Solar Satellite can be added to unit queue 10 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 10, 0); + Carbon::setTestNow($testTime); + + $this->addShipyardBuildRequest('solar_satellite', 1); + + // Verify the building is finished 10 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 20, 0); + Carbon::setTestNow($testTime); + + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertObjectLevelOnPage($response, 'solar_satellite', 1, 'Solar Satellite build job is not finished yet 10 minute after build request issued.'); + } + + /** + * Verify that research requirements are working for unit objects. + * @throws Exception + */ + public function testUnitQueueResearchRequirement(): void + { + // Set the current time to a specific moment for testing + $testTime = Carbon::create(2024, 1, 1, 12, 0, 0); + Carbon::setTestNow($testTime); + + // Add required resources to planet + $this->planetAddResources(new Resources(5000, 5000, 5000, 0)); + + // Assert that build requirements for Light Fighter are not met + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'light_fighter', 'Light Fighter build requirements are met.'); + + // Add Shipyard and Research Lab to build queue + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('robot_factory'); + $this->addFacilitiesBuildRequest('shipyard'); + $this->addFacilitiesBuildRequest('research_lab'); + + // Verify the building is finished 10 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 10, 0); + Carbon::setTestNow($testTime); + + $response = $this->get('/facilities'); + $response->assertStatus(200); + $this->assertObjectLevelOnPage($response, 'research_lab', 1, 'Research Lab build job is not finished yet 10 minute after build request issued.'); + + // Assert that build requirements for Light Fighter are not met as Combustion Drive technology is missing + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'light_fighter', 'Light Fighter build requirements are met.'); + + // Add required technology to research queue + $this->addResearchBuildRequest('energy_technology'); + $this->addResearchBuildRequest('combustion_drive'); + + // Assert that build requirements for Light Fighter are not met as Combustion Drive technology is in build queue + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertRequirementsNotMet($response, 'light_fighter', 'Light Fighter build requirements are met.'); + + // Verify the research is finished 10 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 20, 0); + Carbon::setTestNow($testTime); + + $this->planetService->getPlayer()->updateResearchQueue(); + $response = $this->get('/research'); + $response->assertStatus(200); + $this->assertObjectLevelOnPage($response, 'combustion_drive', 1, 'Combustion Drive is not at level one 10 minutes after build request issued.'); + + $this->addShipyardBuildRequest('light_fighter', 1); + + // Verify the building is finished 10 minute later. + $testTime = Carbon::create(2024, 1, 1, 12, 30, 0); + Carbon::setTestNow($testTime); + + $response = $this->get('/shipyard'); + $response->assertStatus(200); + $this->assertObjectLevelOnPage($response, 'light_fighter', 1, 'Light Fighter build job is not finished yet 10 minute after build request issued.'); + } + /** * Verify that unit construction time is calculated correctly (higher than 0) * @throws BindingResolutionException diff --git a/tests/Unit/BuildingQueueServiceTest.php b/tests/Unit/BuildingQueueServiceTest.php deleted file mode 100644 index fa28bf18..00000000 --- a/tests/Unit/BuildingQueueServiceTest.php +++ /dev/null @@ -1,71 +0,0 @@ -building_queue = new BuildingQueueService($object_service); - } - - /** - * Tests object is found from building queue - */ - public function testIsObjectInBuildingQueue(): void - { - // Add level 3 shipyard to building queue - $queue = new BuildingQueue(); - $queue->planet_id = $this->planetService->getPlanetId(); - $queue->object_id = 21; - $queue->object_level_target = 3; - $queue->save(); - - $this->assertTrue($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 3)); - $this->assertFalse($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 4)); - $this->assertFalse($this->building_queue->objectInBuildingQueue($this->planetService, 'robot_factory', 3)); - } - - /** - * Tests building queue item is cancelled if requirements are not met. - */ - public function testCancelObjectMissingRequirements(): void - { - // Add level 2 robot factory to building queue - $queue_robot_factory = new BuildingQueue(); - $queue_robot_factory->planet_id = $this->planetService->getPlanetId(); - $queue_robot_factory->object_id = 14; - $queue_robot_factory->object_level_target = 2; - $queue_robot_factory->save(); - - // Add level 1 shipyard to building queue - $queue = new BuildingQueue(); - $queue->planet_id = $this->planetService->getPlanetId(); - $queue->object_id = 21; - $queue->object_level_target = 1; - $queue->save(); - - // Assert that shipyard is in building queue - $this->assertTrue($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 1)); - - // Cancel robot factory - $this->building_queue->cancel($this->planetService, $queue_robot_factory->id, 14); - $this->building_queue->cancelItemMissingRequirements($this->planetService); - - // Assert that shipyard is in building queue - $this->assertFalse($this->building_queue->objectInBuildingQueue($this->planetService, 'shipyard', 1)); - } -} diff --git a/tests/Unit/ObjectServiceTest.php b/tests/Unit/ObjectServiceTest.php index e104b6f1..575f6040 100644 --- a/tests/Unit/ObjectServiceTest.php +++ b/tests/Unit/ObjectServiceTest.php @@ -2,77 +2,42 @@ namespace Tests\Unit; -use OGame\Models\BuildingQueue; -use OGame\Models\Resources; use OGame\Services\ObjectService; -use Tests\AccountTestCase; +use Tests\UnitTestCase; -class ObjectServiceTest extends AccountTestCase +class ObjectServiceTest extends UnitTestCase { - protected ObjectService $object_service; - - /** - * Set up common test components. - */ - protected function setUp(): void - { - parent::setUp(); - - $this->object_service = new ObjectService(); - } - /** * Tests maximum building amount returns correct value. */ public function testGetObjectMaxBuildAmount(): void { + $object_service = new ObjectService(); + $this->createAndSetPlanetModel([]); + // Test with requirements not met - $max_build_amount = $this->object_service->getObjectMaxBuildAmount('plasma_turret', $this->planetService, false); + $max_build_amount = $object_service->getObjectMaxBuildAmount('plasma_turret', $this->planetService, false); $this->assertEquals(0, $max_build_amount); - // Test with object limited to one instance - $max_build_amount = $this->object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); + // Test with object limited to one instance per user + $max_build_amount = $object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); $this->assertEquals(1, $max_build_amount); + $this->createAndSetPlanetModel([ + 'small_shield_dome' => 1, + ]); + // Test with object limited to one instance which already exists - $this->planetSetObjectLevel('small_shield_dome', 1); - $max_build_amount = $this->object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); + $max_build_amount = $object_service->getObjectMaxBuildAmount('small_shield_dome', $this->planetService, true); $this->assertEquals(0, $max_build_amount); + $this->createAndSetPlanetModel([ + 'metal' => 24000, + 'crystal' => 6000 + ]); + // Test it calculates max amount correctly - $this->planetAddResources(new Resources(24000, 6000, 0, 0)); - $max_build_amount = $this->object_service->getObjectMaxBuildAmount('anti_ballistic_missile', $this->planetService, true); + $max_build_amount = $object_service->getObjectMaxBuildAmount('anti_ballistic_missile', $this->planetService, true); $this->assertEquals(3, $max_build_amount); } - - /** - * Tests object requirements are verified against prior levels, research and buildings - * including items in building and research queues. - */ - public function testObjectRequirementsMet(): void - { - // Assert that requirements are not met if prior levels doesn't exist - $this->assertFalse($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->planetService->getPlayer(), 2)); - - $this->planetSetObjectLevel('robot_factory', 1); - - // Assert that requirements are met if prior levels exists - $this->assertTrue($this->object_service->objectRequirementsMet('robot_factory', $this->planetService, $this->planetService->getPlayer(), 2)); - - // Assert that requirements are not met if requisites are missing - $this->assertFalse($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->planetService->getPlayer(), 1)); - - // Add to build queue - $queue = new BuildingQueue(); - $queue->planet_id = $this->planetService->getPlanetId(); - $queue->object_id = 21; - $queue->object_level_target = 1; - $queue->save(); - - // Assert that requirements are met if requisites are in build queue - $this->assertTrue($this->object_service->objectRequirementsMet('missile_silo', $this->planetService, $this->planetService->getPlayer(), 1)); - - // Assert that research requirements are not met if building requirements are not met - $this->assertFalse($this->object_service->objectRequirementsMet('computer_technology', $this->planetService, $this->planetService->getPlayer(), 1)); - } } diff --git a/tests/Unit/PlanetServiceTest.php b/tests/Unit/PlanetServiceTest.php index c58515e9..5c143fda 100644 --- a/tests/Unit/PlanetServiceTest.php +++ b/tests/Unit/PlanetServiceTest.php @@ -5,7 +5,6 @@ use Illuminate\Contracts\Container\BindingResolutionException; use OGame\Models\BuildingQueue; use OGame\Models\Enums\ResourceType; -use OGame\Models\Planet; use OGame\Models\Resources; use Tests\UnitTestCase; @@ -200,8 +199,9 @@ public function testGetPlanetBuildingCount(): void */ public function testIsBuildingObject(): void { - $planet = Planet::factory()->make(['id' => 1]); - $this->planetService->setPlanet($planet); + $this->createAndSetPlanetModel([ + 'id' => 1, + ]); // Add level 3 shipyard to building queue $queue = new BuildingQueue(); @@ -211,7 +211,5 @@ public function testIsBuildingObject(): void $queue->save(); $this->assertTrue($this->planetService->isBuildingObject('shipyard', 3)); - $this->assertFalse($this->planetService->isBuildingObject('shipyard', 4)); - $this->assertFalse($this->planetService->isBuildingObject('robot_factory', 3)); } } diff --git a/tests/Unit/ResearchQueueServiceTest.php b/tests/Unit/ResearchQueueServiceTest.php deleted file mode 100644 index f7888d20..00000000 --- a/tests/Unit/ResearchQueueServiceTest.php +++ /dev/null @@ -1,71 +0,0 @@ -research_queue = new ResearchQueueService($object_service); - } - - /** - * Tests object is found from research queue - */ - public function testIsObjectInResearchQueue(): void - { - // Add level 3 impulse drive to research queue - $queue = new ResearchQueue(); - $queue->planet_id = $this->planetService->getPlanetId(); - $queue->object_id = 117; - $queue->object_level_target = 3; - $queue->save(); - - $this->assertTrue($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 3)); - $this->assertFalse($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 4)); - $this->assertFalse($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'energy_technology', 4)); - } - - /** - * Tests research queue item is cancelled if requirements are not met. - */ - public function testCancelItemMissingRequirements(): void - { - // Add level 1 energy technology to research queue - $queue_energy_tech = new ResearchQueue(); - $queue_energy_tech->planet_id = $this->planetService->getPlanetId(); - $queue_energy_tech->object_id = 113; - $queue_energy_tech->object_level_target = 1; - $queue_energy_tech->save(); - - // Add level 1 impulse drive to research queue - $queue = new ResearchQueue(); - $queue->planet_id = $this->planetService->getPlanetId(); - $queue->object_id = 117; - $queue->object_level_target = 1; - $queue->save(); - - // Assert that impulse drive is in research queue - $this->assertTrue($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 1)); - - // Cancel energy technology - $this->research_queue->cancel($this->planetService->getPlayer(), $queue_energy_tech->id, 113); - $this->research_queue->cancelItemMissingRequirements($this->planetService->getPlayer(), $this->planetService); - - // Assert that impulse drive is in research queue - $this->assertFalse($this->research_queue->objectInResearchQueue($this->planetService->getPlayer(), 'impulse_drive', 1)); - } -}