mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-05-18 19:55:33 +00:00
261 lines
7.4 KiB
PHP
261 lines
7.4 KiB
PHP
<?php
|
|
|
|
/*
|
|
*
|
|
* ____ _ _ __ __ _ __ __ ____
|
|
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
|
|
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
|
|
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
|
|
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* @author PocketMine Team
|
|
* @link http://www.pocketmine.net/
|
|
*
|
|
*
|
|
*/
|
|
|
|
namespace pocketmine\network;
|
|
|
|
use pocketmine\event\player\PlayerCreationEvent;
|
|
use pocketmine\network\protocol\DataPacket;
|
|
use pocketmine\network\protocol\Info as ProtocolInfo;
|
|
use pocketmine\network\protocol\UnknownPacket;
|
|
use pocketmine\Player;
|
|
use pocketmine\Server;
|
|
use pocketmine\utils\MainLogger;
|
|
use raklib\protocol\EncapsulatedPacket;
|
|
use raklib\RakLib;
|
|
use raklib\server\RakLibServer;
|
|
use raklib\server\ServerHandler;
|
|
use raklib\server\ServerInstance;
|
|
|
|
class RakLibInterface implements ServerInstance, AdvancedSourceInterface{
|
|
|
|
/** @var Server */
|
|
private $server;
|
|
|
|
/** @var Network */
|
|
private $network;
|
|
|
|
/** @var RakLibServer */
|
|
private $rakLib;
|
|
|
|
/** @var Player[] */
|
|
private $players = [];
|
|
|
|
/** @var \SplObjectStorage */
|
|
private $identifiers;
|
|
|
|
/** @var int[] */
|
|
private $identifiersACK = [];
|
|
|
|
/** @var ServerHandler */
|
|
private $interface;
|
|
|
|
public function __construct(Server $server){
|
|
|
|
$this->server = $server;
|
|
$this->identifiers = new \SplObjectStorage();
|
|
|
|
$this->rakLib = new RakLibServer($this->server->getLogger(), $this->server->getLoader(), $this->server->getPort(), $this->server->getIp() === "" ? "0.0.0.0" : $this->server->getIp());
|
|
$this->interface = new ServerHandler($this->rakLib, $this);
|
|
|
|
for($i = 0; $i < 256; ++$i){
|
|
$this->channelCounts[$i] = 0;
|
|
}
|
|
}
|
|
|
|
public function setNetwork(Network $network){
|
|
$this->network = $network;
|
|
}
|
|
|
|
public function doTick(){
|
|
if(!$this->rakLib->isTerminated()){
|
|
$this->interface->sendTick();
|
|
}else{
|
|
$info = $this->rakLib->getTerminationInfo();
|
|
$this->network->unregisterInterface($this);
|
|
\ExceptionHandler::handler(E_ERROR, "RakLib Thread crashed [".$info["scope"]."]: " . (isset($info["message"]) ? $info["message"] : ""), $info["file"], $info["line"]);
|
|
}
|
|
}
|
|
|
|
public function process(){
|
|
$work = false;
|
|
if($this->interface->handlePacket()){
|
|
$work = true;
|
|
while($this->interface->handlePacket()){
|
|
}
|
|
}
|
|
|
|
$this->doTick();
|
|
|
|
return $work;
|
|
}
|
|
|
|
public function closeSession($identifier, $reason){
|
|
if(isset($this->players[$identifier])){
|
|
$player = $this->players[$identifier];
|
|
$this->identifiers->detach($player);
|
|
unset($this->players[$identifier]);
|
|
unset($this->identifiersACK[$identifier]);
|
|
if(!$player->closed){
|
|
$player->close($player->getLeaveMessage(), $reason);
|
|
}
|
|
}
|
|
}
|
|
|
|
public function close(Player $player, $reason = "unknown reason"){
|
|
if(isset($this->identifiers[$player])){
|
|
unset($this->players[$this->identifiers[$player]]);
|
|
unset($this->identifiersACK[$this->identifiers[$player]]);
|
|
$this->interface->closeSession($this->identifiers[$player], $reason);
|
|
$this->identifiers->detach($player);
|
|
}
|
|
}
|
|
|
|
public function shutdown(){
|
|
$this->interface->shutdown();
|
|
}
|
|
|
|
public function emergencyShutdown(){
|
|
$this->interface->emergencyShutdown();
|
|
}
|
|
|
|
public function openSession($identifier, $address, $port, $clientID){
|
|
$ev = new PlayerCreationEvent($this, Player::class, Player::class, null, $address, $port);
|
|
$this->server->getPluginManager()->callEvent($ev);
|
|
$class = $ev->getPlayerClass();
|
|
|
|
$player = new $class($this, $ev->getClientId(), $ev->getAddress(), $ev->getPort());
|
|
$this->players[$identifier] = $player;
|
|
$this->identifiersACK[$identifier] = 0;
|
|
$this->identifiers->attach($player, $identifier);
|
|
$this->server->addPlayer($identifier, $player);
|
|
}
|
|
|
|
public function handleEncapsulated($identifier, EncapsulatedPacket $packet, $flags){
|
|
if(isset($this->players[$identifier])){
|
|
try{
|
|
if($packet->buffer !== ""){
|
|
$pk = $this->getPacket($packet->buffer);
|
|
$pk->decode();
|
|
$this->players[$identifier]->handleDataPacket($pk);
|
|
}
|
|
}catch(\Exception $e){
|
|
if(\pocketmine\DEBUG > 1 and isset($pk)){
|
|
$logger = $this->server->getLogger();
|
|
if($logger instanceof MainLogger){
|
|
$logger->debug("Packet " . get_class($pk) . " 0x" . bin2hex($packet->buffer));
|
|
$logger->logException($e);
|
|
}
|
|
}
|
|
|
|
if(isset($this->players[$identifier])){
|
|
$this->interface->blockAddress($this->players[$identifier]->getAddress(), 5);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public function blockAddress($address, $timeout = 300){
|
|
$this->interface->blockAddress($address, $timeout);
|
|
}
|
|
|
|
public function handleRaw($address, $port, $payload){
|
|
$this->server->handlePacket($address, $port, $payload);
|
|
}
|
|
|
|
public function sendRawPacket($address, $port, $payload){
|
|
$this->interface->sendRaw($address, $port, $payload);
|
|
}
|
|
|
|
public function notifyACK($identifier, $identifierACK){
|
|
|
|
}
|
|
|
|
public function setName($name){
|
|
$this->interface->sendOption("name", "MCCPP;Demo;$name");
|
|
}
|
|
|
|
public function setPortCheck($name){
|
|
$this->interface->sendOption("portChecking", (bool) $name);
|
|
}
|
|
|
|
public function handleOption($name, $value){
|
|
if($name === "bandwidth"){
|
|
$v = unserialize($value);
|
|
$this->network->addStatistics($v["up"], $v["down"]);
|
|
}
|
|
}
|
|
|
|
public function putPacket(Player $player, DataPacket $packet, $needACK = false, $immediate = false){
|
|
if(isset($this->identifiers[$player])){
|
|
$identifier = $this->identifiers[$player];
|
|
$pk = null;
|
|
if(!$packet->isEncoded){
|
|
$packet->encode();
|
|
}elseif(!$needACK){
|
|
if(!isset($packet->__encapsulatedPacket)){
|
|
$packet->__encapsulatedPacket = new CachedEncapsulatedPacket;
|
|
$packet->__encapsulatedPacket->identifierACK = null;
|
|
$packet->__encapsulatedPacket->buffer = $packet->buffer;
|
|
if($packet->getChannel() !== 0){
|
|
$packet->__encapsulatedPacket->reliability = 3;
|
|
$packet->__encapsulatedPacket->orderChannel = $packet->getChannel();
|
|
$packet->__encapsulatedPacket->orderIndex = 0;
|
|
}else{
|
|
$packet->__encapsulatedPacket->reliability = 2;
|
|
}
|
|
}
|
|
$pk = $packet->__encapsulatedPacket;
|
|
}
|
|
|
|
if(!$immediate and !$needACK and $packet->pid() !== ProtocolInfo::BATCH_PACKET
|
|
and Network::$BATCH_THRESHOLD >= 0
|
|
and strlen($packet->buffer) >= Network::$BATCH_THRESHOLD){
|
|
$this->server->batchPackets([$player], [$packet], true, $packet->getChannel());
|
|
return null;
|
|
}
|
|
|
|
if($pk === null){
|
|
$pk = new EncapsulatedPacket();
|
|
$pk->buffer = $packet->buffer;
|
|
if($packet->getChannel() !== 0){
|
|
$packet->reliability = 3;
|
|
$packet->orderChannel = $packet->getChannel();
|
|
$packet->orderIndex = 0;
|
|
}else{
|
|
$packet->reliability = 2;
|
|
}
|
|
|
|
if($needACK === true){
|
|
$pk->identifierACK = $this->identifiersACK[$identifier]++;
|
|
}
|
|
}
|
|
|
|
$this->interface->sendEncapsulated($identifier, $pk, ($needACK === true ? RakLib::FLAG_NEED_ACK : 0) | ($immediate === true ? RakLib::PRIORITY_IMMEDIATE : RakLib::PRIORITY_NORMAL));
|
|
|
|
return $pk->identifierACK;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private function getPacket($buffer){
|
|
$pid = ord($buffer{0});
|
|
|
|
if(($data = $this->network->getPacket($pid)) === null){
|
|
$data = new UnknownPacket();
|
|
$data->packetID = $pid;
|
|
}
|
|
$data->setBuffer($buffer, 1);
|
|
|
|
return $data;
|
|
}
|
|
}
|