allocationRepository = $allocationRepository; $this->daemonServerRepository = $daemonServerRepository; $this->connection = $connection; $this->repository = $repository; $this->structureService = $structureService; } /** * 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\Repository\RecordNotFoundException * @throws \Pterodactyl\Exceptions\Model\DataValidationException */ 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'), 'threads' => array_get($data, 'threads'), 'disk' => array_get($data, 'disk'), 'allocation_id' => array_get($data, 'allocation_id'), 'database_limit' => array_get($data, 'database_limit', 0) ?? null, 'allocation_limit' => array_get($data, 'allocation_limit', 0) ?? null, 'backup_limit' => array_get($data, 'backup_limit', 0) ?? null, ]); $updateData = $this->structureService->handle($server); try { $this->daemonServerRepository ->setServer($server) ->update(Arr::only($updateData, ['build'])); $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 = $server->allocations->pluck('id')->toArray(); $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]); } } } }