server = $server; $this->rakServerId = mt_rand(0, PHP_INT_MAX); $this->sleeper = new SleeperNotifier(); $mainToThreadBuffer = new \Threaded; $threadToMainBuffer = new \Threaded; $this->rakLib = new RakLibServer( $this->server->getLogger(), $mainToThreadBuffer, $threadToMainBuffer, new InternetAddress($this->server->getIp(), $this->server->getPort(), 4), $this->rakServerId, (int) $this->server->getProperty("network.max-mtu-size", 1492), self::MCPE_RAKNET_PROTOCOL_VERSION, $this->sleeper ); $this->eventReceiver = new RakLibToUserThreadMessageReceiver( $this, new InterThreadChannelReader($threadToMainBuffer) ); $this->interface = new UserToRakLibThreadMessageSender( new InterThreadChannelWriter($mainToThreadBuffer) ); } public function start() : void{ $this->server->getTickSleeper()->addNotifier($this->sleeper, function() : void{ while($this->eventReceiver->handle()); }); $this->server->getLogger()->debug("Waiting for RakLib to start..."); $this->rakLib->startAndWait(PTHREADS_INHERIT_CONSTANTS); //HACK: MainLogger needs constants for exception logging $this->server->getLogger()->debug("RakLib booted successfully"); } public function setNetwork(Network $network) : void{ $this->network = $network; } public function tick() : void{ if(!$this->rakLib->isRunning()){ $e = $this->rakLib->getCrashInfo(); if($e !== null){ throw $e; } throw new \Exception("RakLib Thread crashed without crash information"); } } public function closeSession(int $sessionId, string $reason) : void{ if(isset($this->sessions[$sessionId])){ $session = $this->sessions[$sessionId]; unset($this->sessions[$sessionId]); $session->onClientDisconnect($reason); } } public function close(int $sessionId) : void{ if(isset($this->sessions[$sessionId])){ unset($this->sessions[$sessionId]); $this->interface->closeSession($sessionId); } } public function shutdown() : void{ $this->server->getTickSleeper()->removeNotifier($this->sleeper); $this->interface->shutdown(); $this->rakLib->quit(); } public function openSession(int $sessionId, string $address, int $port, int $clientID) : void{ $session = new NetworkSession($this->server, $this->network->getSessionManager(), new RakLibPacketSender($sessionId, $this), $address, $port); $this->sessions[$sessionId] = $session; } public function handleEncapsulated(int $sessionId, string $packet) : void{ if(isset($this->sessions[$sessionId])){ if($packet === "" or $packet[0] !== self::MCPE_RAKNET_PACKET_ID){ return; } //get this now for blocking in case the player was closed before the exception was raised $session = $this->sessions[$sessionId]; $address = $session->getIp(); $buf = substr($packet, 1); try{ $session->handleEncoded($buf); }catch(BadPacketException $e){ $errorId = bin2hex(random_bytes(6)); $logger = $session->getLogger(); $logger->error("Bad packet (error ID $errorId): " . $e->getMessage()); //intentionally doesn't use logException, we don't want spammy packet error traces to appear in release mode $logger->debug("Origin: " . Filesystem::cleanPath($e->getFile()) . "(" . $e->getLine() . ")"); foreach(Utils::printableTrace($e->getTrace()) as $frame){ $logger->debug($frame); } $session->disconnect("Packet processing error (Error ID: $errorId)"); $this->interface->blockAddress($address, 5); } } } public function blockAddress(string $address, int $timeout = 300) : void{ $this->interface->blockAddress($address, $timeout); } public function unblockAddress(string $address) : void{ $this->interface->unblockAddress($address); } public function handleRaw(string $address, int $port, string $payload) : void{ $this->network->processRawPacket($this, $address, $port, $payload); } public function sendRawPacket(string $address, int $port, string $payload) : void{ $this->interface->sendRaw($address, $port, $payload); } public function addRawPacketFilter(string $regex) : void{ $this->interface->addRawPacketFilter($regex); } public function notifyACK(int $sessionId, int $identifierACK) : void{ } public function setName(string $name) : void{ $info = $this->server->getQueryInformation(); $this->interface->setOption("name", implode(";", [ "MCPE", rtrim(addcslashes($name, ";"), '\\'), ProtocolInfo::CURRENT_PROTOCOL, ProtocolInfo::MINECRAFT_VERSION_NETWORK, $info->getPlayerCount(), $info->getMaxPlayerCount(), $this->rakServerId, $this->server->getName(), $this->server->getGamemode()->getEnglishName() ]) . ";" ); } public function setPortCheck(bool $name) : void{ $this->interface->setOption("portChecking", $name); } public function handleOption(string $option, string $value) : void{ if($option === "bandwidth"){ $v = unserialize($value); $this->network->addStatistics($v["up"], $v["down"]); } } public function putPacket(int $sessionId, string $payload, bool $immediate = true) : void{ if(isset($this->sessions[$sessionId])){ $pk = new EncapsulatedPacket(); $pk->buffer = self::MCPE_RAKNET_PACKET_ID . $payload; $pk->reliability = PacketReliability::RELIABLE_ORDERED; $pk->orderChannel = 0; $this->interface->sendEncapsulated($sessionId, $pk, ($immediate ? RakLib::PRIORITY_IMMEDIATE : RakLib::PRIORITY_NORMAL)); } } public function updatePing(int $sessionId, int $pingMS) : void{ if(isset($this->sessions[$sessionId])){ $this->sessions[$sessionId]->updatePing($pingMS); } } }