74 private \Closure $playerInfoConsumer,
75 private \Closure $authCallback
78 private static function calculateUuidFromXuid(
string $xuid) : UuidInterface{
79 $hash = md5(
"pocket-auth-1-xuid:" . $xuid, binary: true);
80 $hash[6] = chr((ord($hash[6]) & 0x0f) | 0x30);
81 $hash[8] = chr((ord($hash[8]) & 0x3f) | 0x80);
83 return Uuid::fromBytes($hash);
86 public function handleLogin(LoginPacket $packet) : bool{
87 $authInfo = $this->parseAuthInfo($packet->authInfoJson);
89 if($authInfo->AuthenticationType === AuthenticationType::FULL->value){
91 [$headerArray, $claimsArray,] = JwtUtils::parse($authInfo->Token);
93 throw PacketHandlingException::wrap($e,
"Error parsing authentication token");
95 $header = $this->mapXboxTokenHeader($headerArray);
96 $claims = $this->mapXboxTokenBody($claimsArray);
98 $legacyUuid = self::calculateUuidFromXuid($claims->xid);
99 $username = $claims->xname;
100 $xuid = $claims->xid;
102 $authRequired = $this->processLoginCommon($packet, $username, $legacyUuid, $xuid);
103 if($authRequired ===
null){
107 $this->processOpenIdLogin($authInfo->Token, $header->kid, $packet->clientDataJwt, $authRequired);
109 }elseif($authInfo->AuthenticationType === AuthenticationType::SELF_SIGNED->value){
111 $chainData = json_decode($authInfo->Certificate, flags: JSON_THROW_ON_ERROR);
112 }
catch(\JsonException $e){
113 throw PacketHandlingException::wrap($e,
"Error parsing self-signed certificate chain");
115 if(!is_object($chainData)){
116 throw new PacketHandlingException(
"Unexpected type for self-signed certificate chain: " . gettype($chainData) .
", expected object");
119 $chain = $this->defaultJsonMapper(
"Self-signed auth chain JSON")->map($chainData,
new LegacyAuthChain());
120 }
catch(\JsonMapper_Exception $e){
121 throw PacketHandlingException::wrap($e,
"Error mapping self-signed certificate chain");
123 if(count($chain->chain) > 1 || !isset($chain->chain[0])){
124 throw new PacketHandlingException(
"Expected exactly one certificate in self-signed certificate chain, got " . count($chain->chain));
128 [, $claimsArray, ] = JwtUtils::parse($chain->chain[0]);
129 }
catch(JwtException $e){
130 throw PacketHandlingException::wrap($e,
"Error parsing self-signed certificate");
132 if(!isset($claimsArray[
"extraData"]) || !is_array($claimsArray[
"extraData"])){
133 throw new PacketHandlingException(
"Expected \"extraData\" to be present in self-signed certificate");
137 $claims = $this->defaultJsonMapper(
"Self-signed auth JWT 'extraData'")->map($claimsArray[
"extraData"],
new LegacyAuthIdentityData());
138 }
catch(\JsonMapper_Exception $e){
139 throw PacketHandlingException::wrap($e,
"Error mapping self-signed certificate extraData");
142 if(!Uuid::isValid($claims->identity)){
143 throw new PacketHandlingException(
"Invalid UUID string in self-signed certificate: " . $claims->identity);
145 $legacyUuid = Uuid::fromString($claims->identity);
146 $username = $claims->displayName;
149 $authRequired = $this->processLoginCommon($packet, $username, $legacyUuid, $xuid);
150 if($authRequired ===
null){
154 $this->processSelfSignedLogin($chain->chain, $packet->clientDataJwt, $authRequired);
156 throw new PacketHandlingException(
"Unsupported authentication type: $authInfo->AuthenticationType");
162 private function processLoginCommon(LoginPacket $packet,
string $username, UuidInterface $legacyUuid,
string $xuid) : ?bool{
163 if(!Player::isValidUserName($username)){
164 $this->session->disconnectWithError(KnownTranslationFactory::disconnectionScreen_invalidName());
169 $clientData = $this->parseClientData($packet->clientDataJwt);
172 $skin = $this->session->getTypeConverter()->getSkinAdapter()->fromSkinData(ClientDataToSkinDataHelper::fromClientData($clientData));
173 }
catch(\InvalidArgumentException | InvalidSkinException $e){
174 $this->session->disconnectWithError(
175 reason:
"Invalid skin: " . $e->getMessage(),
176 disconnectScreenMessage: KnownTranslationFactory::disconnectionScreen_invalidSkin()
183 $playerInfo =
new XboxLivePlayerInfo(
188 $clientData->LanguageCode,
192 $playerInfo =
new PlayerInfo(
196 $clientData->LanguageCode,
200 ($this->playerInfoConsumer)($playerInfo);
202 $ev =
new PlayerPreLoginEvent(
204 $this->session->getIp(),
205 $this->session->getPort(),
206 $this->server->requiresAuthentication()
208 if($this->
server->getNetwork()->getValidConnectionCount() > $this->server->getMaxPlayers()){
209 $ev->setKickFlag(PlayerPreLoginEvent::KICK_FLAG_SERVER_FULL, KnownTranslationFactory::disconnectionScreen_serverFull());
211 if(!$this->
server->isWhitelisted($playerInfo->getUsername())){
212 $ev->setKickFlag(PlayerPreLoginEvent::KICK_FLAG_SERVER_WHITELISTED, KnownTranslationFactory::pocketmine_disconnect_whitelisted());
216 if(($banEntry = $this->
server->getNameBans()->getEntry($playerInfo->getUsername())) !==
null){
217 $banReason = $banEntry->getReason();
218 $banMessage = $banReason ===
"" ? KnownTranslationFactory::pocketmine_disconnect_ban_noReason() : KnownTranslationFactory::pocketmine_disconnect_ban($banReason);
219 }elseif(($banEntry = $this->
server->getIPBans()->getEntry($this->session->getIp())) !==
null){
220 $banReason = $banEntry->getReason();
221 $banMessage = KnownTranslationFactory::pocketmine_disconnect_ban($banReason !==
"" ? $banReason : KnownTranslationFactory::pocketmine_disconnect_ban_ip());
223 if($banMessage !==
null){
224 $ev->setKickFlag(PlayerPreLoginEvent::KICK_FLAG_BANNED, $banMessage);
228 if(!$ev->isAllowed()){
229 $this->session->disconnect($ev->getFinalDisconnectReason(), $ev->getFinalDisconnectScreenMessage());
233 return $ev->isAuthRequired();
241 $authInfoJson = json_decode($authInfo, associative: false, flags: JSON_THROW_ON_ERROR);
242 }
catch(\JsonException $e){
243 throw PacketHandlingException::wrap($e);
245 if(!is_object($authInfoJson)){
246 throw new PacketHandlingException(
"Unexpected type for auth info data: " . gettype($authInfoJson) .
", expected object");
249 $mapper = $this->defaultJsonMapper(
"Root authentication info JSON");
251 $clientData = $mapper->map($authInfoJson,
new AuthenticationInfo());
252 }
catch(\JsonMapper_Exception $e){
253 throw PacketHandlingException::wrap($e);
263 $mapper = $this->defaultJsonMapper(
"OpenID JWT header");
266 }
catch(\JsonMapper_Exception $e){
267 throw PacketHandlingException::wrap($e);
277 $mapper = $this->defaultJsonMapper(
"OpenID JWT body");
280 }
catch(\JsonMapper_Exception $e){
281 throw PacketHandlingException::wrap($e);
291 [, $clientDataClaims, ] =
JwtUtils::parse($clientDataJwt);
293 throw PacketHandlingException::wrap($e);
296 $mapper = $this->defaultJsonMapper(
"ClientData JWT body");
298 $clientData = $mapper->map($clientDataClaims,
new ClientData());
299 }
catch(\JsonMapper_Exception $e){
300 throw PacketHandlingException::wrap($e);
311 protected function processOpenIdLogin(
string $token,
string $keyId,
string $clientData,
bool $authRequired) : void{
312 $this->session->setHandler(null);
314 $authKeyProvider = $this->
server->getAuthKeyProvider();
316 $authKeyProvider->getKey($keyId)->onCompletion(
317 function(array $issuerAndKey) use ($token, $clientData, $authRequired) :
void{
318 [$issuer, $mojangPublicKeyPem] = $issuerAndKey;
319 $this->
server->getAsyncPool()->submitTask(
new ProcessOpenIdLoginTask($token, $issuer, $mojangPublicKeyPem, $clientData, $authRequired, $this->authCallback));
321 fn() => ($this->authCallback)(
false, $authRequired,
"Unrecognized authentication key ID: $keyId",
null)
329 $this->session->setHandler(null);
331 $this->
server->getAsyncPool()->submitTask(
new ProcessLegacyLoginTask($legacyCertificate, $clientDataJwt, rootAuthKeyDer:
null, authRequired: $authRequired, onCompletion: $this->authCallback));
334 private function defaultJsonMapper(
string $logContext) : \JsonMapper{
335 $mapper = new \JsonMapper();
336 $mapper->bExceptionOnMissingData =
true;
337 $mapper->undefinedPropertyHandler = $this->warnUndefinedJsonPropertyHandler($logContext);
338 $mapper->bStrictObjectTypeChecking =
true;
339 $mapper->bEnforceMapType =
false;
346 private function warnUndefinedJsonPropertyHandler(
string $context) : \Closure{
347 return fn(object $object, string $name, mixed $value) => $this->session->getLogger()->warning(
348 "$context: Unexpected JSON property for " . (new \ReflectionClass($object))->getShortName() .
": " .
Utils::printable(substr($name, 0, 80))