2020-10-31 21:58:15 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Pterodactyl\Services\Allocations;
|
|
|
|
|
|
|
|
use Webmozart\Assert\Assert;
|
|
|
|
use Pterodactyl\Models\Server;
|
|
|
|
use Pterodactyl\Models\Allocation;
|
|
|
|
use Pterodactyl\Exceptions\Service\Allocation\AutoAllocationNotEnabledException;
|
|
|
|
use Pterodactyl\Exceptions\Service\Allocation\NoAutoAllocationSpaceAvailableException;
|
|
|
|
|
|
|
|
class FindAssignableAllocationService
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* FindAssignableAllocationService constructor.
|
|
|
|
*/
|
2022-10-14 17:59:20 +01:00
|
|
|
public function __construct(private AssignmentService $service)
|
2020-10-31 21:58:15 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finds an existing unassigned allocation and attempts to assign it to the given server. If
|
|
|
|
* no allocation can be found, a new one will be created with a random port between the defined
|
|
|
|
* range from the configuration.
|
|
|
|
*
|
|
|
|
* @throws \Pterodactyl\Exceptions\DisplayException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\CidrOutOfRangeException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\InvalidPortMappingException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\PortOutOfRangeException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\TooManyPortsInRangeException
|
|
|
|
*/
|
2022-10-14 17:59:20 +01:00
|
|
|
public function handle(Server $server): Allocation
|
2020-10-31 21:58:15 +00:00
|
|
|
{
|
2021-01-23 20:33:34 +00:00
|
|
|
if (!config('pterodactyl.client_features.allocations.enabled')) {
|
|
|
|
throw new AutoAllocationNotEnabledException();
|
2020-10-31 21:58:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to find a given available allocation for a server. If one cannot be found
|
|
|
|
// we will fall back to attempting to create a new allocation that can be used for the
|
|
|
|
// server.
|
|
|
|
/** @var \Pterodactyl\Models\Allocation|null $allocation */
|
|
|
|
$allocation = $server->node->allocations()
|
|
|
|
->where('ip', $server->allocation->ip)
|
|
|
|
->whereNull('server_id')
|
|
|
|
->inRandomOrder()
|
|
|
|
->first();
|
|
|
|
|
|
|
|
$allocation = $allocation ?? $this->createNewAllocation($server);
|
|
|
|
|
|
|
|
$allocation->update(['server_id' => $server->id]);
|
|
|
|
|
|
|
|
return $allocation->refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-11-01 05:17:32 +00:00
|
|
|
* Create a new allocation on the server's node with a random port from the defined range
|
|
|
|
* in the settings. If there are no matches in that range, or something is wrong with the
|
|
|
|
* range information provided an exception will be raised.
|
|
|
|
*
|
2020-10-31 21:58:15 +00:00
|
|
|
* @throws \Pterodactyl\Exceptions\DisplayException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\CidrOutOfRangeException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\InvalidPortMappingException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\PortOutOfRangeException
|
|
|
|
* @throws \Pterodactyl\Exceptions\Service\Allocation\TooManyPortsInRangeException
|
|
|
|
*/
|
|
|
|
protected function createNewAllocation(Server $server): Allocation
|
|
|
|
{
|
|
|
|
$start = config('pterodactyl.client_features.allocations.range_start', null);
|
|
|
|
$end = config('pterodactyl.client_features.allocations.range_end', null);
|
|
|
|
|
2021-01-23 20:33:34 +00:00
|
|
|
if (!$start || !$end) {
|
|
|
|
throw new NoAutoAllocationSpaceAvailableException();
|
2020-10-31 21:58:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Assert::integerish($start);
|
|
|
|
Assert::integerish($end);
|
|
|
|
|
|
|
|
// Get all of the currently allocated ports for the node so that we can figure out
|
|
|
|
// which port might be available.
|
|
|
|
$ports = $server->node->allocations()
|
|
|
|
->where('ip', $server->allocation->ip)
|
|
|
|
->whereBetween('port', [$start, $end])
|
|
|
|
->pluck('port');
|
|
|
|
|
|
|
|
// Compute the difference of the range and the currently created ports, finding
|
|
|
|
// any port that does not already exist in the database. We will then use this
|
|
|
|
// array of ports to create a new allocation to assign to the server.
|
|
|
|
$available = array_diff(range($start, $end), $ports->toArray());
|
|
|
|
|
|
|
|
// If we've already allocated all of the ports, just abort.
|
|
|
|
if (empty($available)) {
|
2021-01-23 20:33:34 +00:00
|
|
|
throw new NoAutoAllocationSpaceAvailableException();
|
2020-10-31 21:58:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Pick a random port out of the remaining available ports.
|
|
|
|
/** @var int $port */
|
|
|
|
$port = $available[array_rand($available)];
|
|
|
|
|
|
|
|
$this->service->handle($server->node, [
|
|
|
|
'allocation_ip' => $server->allocation->ip,
|
|
|
|
'allocation_ports' => [$port],
|
|
|
|
]);
|
|
|
|
|
|
|
|
/** @var \Pterodactyl\Models\Allocation $allocation */
|
|
|
|
$allocation = $server->node->allocations()
|
|
|
|
->where('ip', $server->allocation->ip)
|
|
|
|
->where('port', $port)
|
|
|
|
->firstOrFail();
|
|
|
|
|
|
|
|
return $allocation;
|
|
|
|
}
|
|
|
|
}
|