mirror of
				https://github.com/pelican-dev/panel.git
				synced 2025-10-31 01:56:52 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			129 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			129 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace App\Services\Servers;
 | |
| 
 | |
| use Illuminate\Support\Arr;
 | |
| use App\Models\Server;
 | |
| use App\Models\Allocation;
 | |
| use Illuminate\Database\ConnectionInterface;
 | |
| use App\Exceptions\DisplayException;
 | |
| use App\Repositories\Daemon\DaemonServerRepository;
 | |
| use Illuminate\Http\Client\ConnectionException;
 | |
| 
 | |
| class BuildModificationService
 | |
| {
 | |
|     /**
 | |
|      * BuildModificationService constructor.
 | |
|      */
 | |
|     public function __construct(
 | |
|         private ConnectionInterface $connection,
 | |
|         private DaemonServerRepository $daemonServerRepository,
 | |
|         private ServerConfigurationStructureService $structureService
 | |
|     ) {}
 | |
| 
 | |
|     /**
 | |
|      * Change the build details for a specified server.
 | |
|      *
 | |
|      * @param  array<string, mixed>  $data
 | |
|      *
 | |
|      * @throws \Throwable
 | |
|      * @throws \App\Exceptions\DisplayException
 | |
|      */
 | |
|     public function handle(Server $server, array $data): Server
 | |
|     {
 | |
|         /** @var \App\Models\Server $server */
 | |
|         $server = $this->connection->transaction(function () use ($server, $data) {
 | |
|             $this->processAllocations($server, $data);
 | |
| 
 | |
|             if (isset($data['allocation_id']) && $data['allocation_id'] != $server->allocation_id) {
 | |
|                 $existingAllocation = $server->allocations()->findOrFail($data['allocation_id']);
 | |
| 
 | |
|                 throw_unless($existingAllocation, new DisplayException('The requested default allocation is not currently assigned to this server.'));
 | |
|             }
 | |
| 
 | |
|             if (!isset($data['oom_killer']) && isset($data['oom_disabled'])) {
 | |
|                 $data['oom_killer'] = !$data['oom_disabled'];
 | |
|             }
 | |
| 
 | |
|             // If any of these values are passed through in the data array go ahead and set them correctly on the server model.
 | |
|             $merge = Arr::only($data, ['oom_killer', 'memory', 'swap', 'io', 'cpu', 'threads', 'disk', 'allocation_id']);
 | |
| 
 | |
|             $server->forceFill(array_merge($merge, [
 | |
|                 'database_limit' => Arr::get($data, 'database_limit', 0) ?? null,
 | |
|                 'allocation_limit' => Arr::get($data, 'allocation_limit', 0) ?? null,
 | |
|                 'backup_limit' => Arr::get($data, 'backup_limit', 0) ?? 0,
 | |
|             ]))->saveOrFail();
 | |
| 
 | |
|             return $server->refresh();
 | |
|         });
 | |
| 
 | |
|         $updateData = $this->structureService->handle($server);
 | |
| 
 | |
|         // Because daemon always fetches an updated configuration from the Panel when booting
 | |
|         // a server this type of exception can be safely "ignored" and just written to the logs.
 | |
|         // Ideally this request succeeds, so we can apply resource modifications on the fly, but
 | |
|         // if it fails we can just continue on as normal.
 | |
|         if (!empty($updateData['build'])) {
 | |
|             try {
 | |
|                 $this->daemonServerRepository->setServer($server)->sync();
 | |
|             } catch (ConnectionException $exception) {
 | |
|                 logger()->warning($exception, ['server_id' => $server->id]);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $server;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Process the allocations being assigned in the data and ensure they are available for a server.
 | |
|      *
 | |
|      * @param array{
 | |
|      *     add_allocations?: array<int>,
 | |
|      *     remove_allocations?: array<int>,
 | |
|      *     allocation_id: ?int,
 | |
|      *     oom_killer?: bool,
 | |
|      *     oom_disabled?: bool,
 | |
|      * } $data
 | |
|      */
 | |
|     private function processAllocations(Server $server, array &$data): void
 | |
|     {
 | |
|         if (empty($data['add_allocations']) && empty($data['remove_allocations'])) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         // Handle the addition of allocations to this server. Only assign allocations that are not currently
 | |
|         // assigned to a different server, and only allocations on the same node as the server.
 | |
|         if (!empty($data['add_allocations'])) {
 | |
|             $query = Allocation::query()
 | |
|                 ->where('node_id', $server->node_id)
 | |
|                 ->whereIn('id', $data['add_allocations'])
 | |
|                 ->whereNull('server_id');
 | |
| 
 | |
|             $query->update(['server_id' => $server->id]);
 | |
|         }
 | |
| 
 | |
|         if (!empty($data['remove_allocations'])) {
 | |
|             $allocations = Allocation::query()
 | |
|                 ->where('server_id', $server->id)
 | |
|                 // Only use the allocations that we didn't also attempt to add to the server...
 | |
|                 ->whereIn('id', array_diff($data['remove_allocations'], $data['add_allocations'] ?? []));
 | |
| 
 | |
|             // If we are attempting to remove the default allocation for the server, see if we can reassign
 | |
|             // to the first provided value in add_allocations.
 | |
|             if ((clone $allocations)->where('id', $server->allocation_id)->exists()) {
 | |
|                 $nonPrimaryAllocations = $server->allocations->whereNotIn('id', $data['remove_allocations']);
 | |
|                 $data['allocation_id'] = $nonPrimaryAllocations->first()->id ?? ($data['add_allocations'][0] ?? null);
 | |
|             }
 | |
| 
 | |
|             // Remove any of the allocations we got that are currently assigned to this server on
 | |
|             // this node. Also set the notes to null, otherwise when re-allocated to a new server those
 | |
|             // notes will be carried over.
 | |
|             $allocations
 | |
|                 ->update([
 | |
|                     'notes' => null,
 | |
|                     'server_id' => null,
 | |
|                 ]);
 | |
|         }
 | |
|     }
 | |
| }
 | 
