allocationRepository = $allocationRepository; $this->daemonServerRepository = $daemonServerRepository; $this->connection = $connection; $this->repository = $repository; } /** * Change the build details for a specified server. * * @param \Pterodactyl\Models\Server $server * @param array $data * @return \Pterodactyl\Models\Server * * @throws \Pterodactyl\Exceptions\DisplayException * @throws \Pterodactyl\Exceptions\Model\DataValidationException * @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException */ public function handle(Server $server, array $data) { $this->connection->beginTransaction(); $this->processAllocations($server, $data); if (isset($data['allocation_id']) && $data['allocation_id'] != $server->allocation_id) { try { $this->allocationRepository->findFirstWhere([ ['id', '=', $data['allocation_id']], ['server_id', '=', $server->id], ]); } catch (RecordNotFoundException $ex) { throw new DisplayException(trans('admin/server.exceptions.default_allocation_not_found')); } } /** @var \Pterodactyl\Models\Server $server */ $server = $this->repository->withFreshModel()->update($server->id, [ 'oom_disabled' => array_get($data, 'oom_disabled'), 'memory' => array_get($data, 'memory'), 'swap' => array_get($data, 'swap'), 'io' => array_get($data, 'io'), 'cpu' => array_get($data, 'cpu'), 'disk' => array_get($data, 'disk'), 'allocation_id' => array_get($data, 'allocation_id'), 'database_limit' => array_get($data, 'database_limit'), 'allocation_limit' => array_get($data, 'allocation_limit'), ]); $updateData = [ 'allocations' => [ 'default' => [ 'ip' => $server->allocation->ip, 'port' => $server->allocation->port, ], 'mappings' => $server->getAllocationMappings(), ], 'build' => [ 'memory' => $server->memory, 'swap' => $server->swap, 'io' => $server->io, 'cpu' => $server->cpu, 'disk' => $server->disk, ], 'container' => [ 'oom_disabled' => $server->oom_disabled, ], ]; try { $this->daemonServerRepository->setServer($server)->update($updateData); $this->connection->commit(); } catch (RequestException $exception) { throw new DaemonConnectionException($exception); } return $server; } /** * Process the allocations being assigned in the data and ensure they * are available for a server. * * @param \Pterodactyl\Models\Server $server * @param array $data * * @throws \Pterodactyl\Exceptions\DisplayException */ private function processAllocations(Server $server, array &$data) { $firstAllocationId = null; if (! array_key_exists('add_allocations', $data) && ! array_key_exists('remove_allocations', $data)) { return; } // Handle the addition of allocations to this server. if (array_key_exists('add_allocations', $data) && ! empty($data['add_allocations'])) { $unassigned = $this->allocationRepository->getUnassignedAllocationIds($server->node_id); $updateIds = []; foreach ($data['add_allocations'] as $allocation) { if (! in_array($allocation, $unassigned)) { continue; } $firstAllocationId = $firstAllocationId ?? $allocation; $updateIds[] = $allocation; } if (! empty($updateIds)) { $this->allocationRepository->updateWhereIn('id', $updateIds, ['server_id' => $server->id]); } } // Handle removal of allocations from this server. if (array_key_exists('remove_allocations', $data) && ! empty($data['remove_allocations'])) { $assigned = $this->allocationRepository->getAssignedAllocationIds($server->id); $updateIds = []; foreach ($data['remove_allocations'] as $allocation) { if (! in_array($allocation, $assigned)) { continue; } if ($allocation == $data['allocation_id']) { if (is_null($firstAllocationId)) { throw new DisplayException(trans('admin/server.exceptions.no_new_default_allocation')); } $data['allocation_id'] = $firstAllocationId; } $updateIds[] = $allocation; } if (! empty($updateIds)) { $this->allocationRepository->updateWhereIn('id', $updateIds, ['server_id' => null]); } } } }