connection = $connection; $this->repository = $repository; } /** * Insert allocations into the database and link them to a specific node. * * @param \Pterodactyl\Models\Node $node * @param array $data * * @throws \Pterodactyl\Exceptions\Service\Allocation\CidrOutOfRangeException * @throws \Pterodactyl\Exceptions\Service\Allocation\PortOutOfRangeException * @throws \Pterodactyl\Exceptions\Service\Allocation\InvalidPortMappingException * @throws \Pterodactyl\Exceptions\Service\Allocation\TooManyPortsInRangeException */ public function handle(Node $node, array $data) { $explode = explode('/', $data['allocation_ip']); if (count($explode) !== 1) { if (! ctype_digit($explode[1]) || ($explode[1] > self::CIDR_MIN_BITS || $explode[1] < self::CIDR_MAX_BITS)) { throw new CidrOutOfRangeException; } } $this->connection->beginTransaction(); foreach (Network::parse(gethostbyname($data['allocation_ip'])) as $ip) { foreach ($data['allocation_ports'] as $port) { if (! is_digit($port) && ! preg_match(self::PORT_RANGE_REGEX, $port)) { throw new InvalidPortMappingException($port); } $insertData = []; if (preg_match(self::PORT_RANGE_REGEX, $port, $matches)) { $block = range($matches[1], $matches[2]); if (count($block) > self::PORT_RANGE_LIMIT) { throw new TooManyPortsInRangeException; } if ((int) $matches[1] <= self::PORT_FLOOR || (int) $matches[2] > self::PORT_CEIL) { throw new PortOutOfRangeException; } foreach ($block as $unit) { $insertData[] = [ 'node_id' => $node->id, 'ip' => $ip->__toString(), 'port' => (int) $unit, 'ip_alias' => array_get($data, 'allocation_alias'), 'server_id' => null, ]; } } else { if ((int) $port <= self::PORT_FLOOR || (int) $port > self::PORT_CEIL) { throw new PortOutOfRangeException; } $insertData[] = [ 'node_id' => $node->id, 'ip' => $ip->__toString(), 'port' => (int) $port, 'ip_alias' => array_get($data, 'allocation_alias'), 'server_id' => null, ]; } $this->repository->insertIgnore($insertData); } } $this->connection->commit(); } }