mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-05-23 22:14:41 +00:00
258 lines
6.2 KiB
PHP
258 lines
6.2 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/
|
|
*
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* Named Binary Tag handling classes
|
|
*/
|
|
namespace pocketmine\nbt;
|
|
|
|
use pocketmine\nbt\tag\Byte;
|
|
use pocketmine\nbt\tag\ByteArray;
|
|
use pocketmine\nbt\tag\Compound;
|
|
use pocketmine\nbt\tag\Double;
|
|
use pocketmine\nbt\tag\End;
|
|
use pocketmine\nbt\tag\Enum;
|
|
use pocketmine\nbt\tag\Float;
|
|
use pocketmine\nbt\tag\Int;
|
|
use pocketmine\nbt\tag\IntArray;
|
|
use pocketmine\nbt\tag\Long;
|
|
use pocketmine\nbt\tag\NamedTAG;
|
|
use pocketmine\nbt\tag\Short;
|
|
use pocketmine\nbt\tag\String;
|
|
use pocketmine\nbt\tag\Tag;
|
|
use pocketmine\utils\Binary;
|
|
|
|
/**
|
|
* Named Binary Tag encoder/decoder
|
|
*/
|
|
class NBT{
|
|
|
|
const LITTLE_ENDIAN = 0;
|
|
const BIG_ENDIAN = 1;
|
|
const TAG_End = 0;
|
|
const TAG_Byte = 1;
|
|
const TAG_Short = 2;
|
|
const TAG_Int = 3;
|
|
const TAG_Long = 4;
|
|
const TAG_Float = 5;
|
|
const TAG_Double = 6;
|
|
const TAG_ByteArray = 7;
|
|
const TAG_String = 8;
|
|
const TAG_Enum = 9;
|
|
const TAG_Compound = 10;
|
|
const TAG_IntArray = 11;
|
|
|
|
private $buffer;
|
|
private $offset;
|
|
private $endianness;
|
|
private $data;
|
|
|
|
public function get($len){
|
|
if($len < 0){
|
|
$this->offset = strlen($this->buffer) - 1;
|
|
|
|
return "";
|
|
}elseif($len === true){
|
|
return substr($this->buffer, $this->offset);
|
|
}
|
|
if($len > 1024){
|
|
return substr($this->buffer, ($this->offset += $len) - $len, $len);
|
|
}
|
|
$buffer = "";
|
|
for(; $len > 0; --$len, ++$this->offset){
|
|
$buffer .= @$this->buffer{$this->offset};
|
|
}
|
|
|
|
return $buffer;
|
|
}
|
|
|
|
public function put($v){
|
|
$this->buffer .= $v;
|
|
}
|
|
|
|
public function feof(){
|
|
return !isset($this->buffer{$this->offset});
|
|
}
|
|
|
|
public function __construct($endianness = self::LITTLE_ENDIAN){
|
|
$this->offset = 0;
|
|
$this->endianness = $endianness & 0x01;
|
|
}
|
|
|
|
public function read($buffer){
|
|
$this->offset = 0;
|
|
$this->buffer = $buffer;
|
|
$this->data = $this->readTag();
|
|
$this->buffer = "";
|
|
}
|
|
|
|
public function readCompressed($buffer, $compression = ZLIB_ENCODING_GZIP){
|
|
$this->read(zlib_decode($buffer));
|
|
}
|
|
|
|
public function write(){
|
|
$this->offset = 0;
|
|
if($this->data instanceof Compound){
|
|
$this->writeTag($this->data);
|
|
|
|
return $this->buffer;
|
|
}else{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public function writeCompressed($compression = ZLIB_ENCODING_GZIP, $level = 7){
|
|
if(($write = $this->write()) !== false){
|
|
return zlib_encode($write, $compression, $level);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public function readTag(){
|
|
switch($this->getByte()){
|
|
case NBT::TAG_Byte:
|
|
$tag = new Byte($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Short:
|
|
$tag = new Short($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Int:
|
|
$tag = new Int($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Long:
|
|
$tag = new Long($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Float:
|
|
$tag = new Float($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Double:
|
|
$tag = new Double($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_ByteArray:
|
|
$tag = new ByteArray($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_String:
|
|
$tag = new String($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Enum:
|
|
$tag = new Enum($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_Compound:
|
|
$tag = new Compound($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
case NBT::TAG_IntArray:
|
|
$tag = new IntArray($this->getString());
|
|
$tag->read($this);
|
|
break;
|
|
|
|
case NBT::TAG_End: //No named tag
|
|
default:
|
|
$tag = new End;
|
|
break;
|
|
}
|
|
return $tag;
|
|
}
|
|
|
|
public function writeTag(Tag $tag){
|
|
$this->putByte($tag->getType());
|
|
if($tag instanceof NamedTAG){
|
|
$this->putString($tag->getName());
|
|
}
|
|
$tag->write($this);
|
|
}
|
|
|
|
public function getByte($signed = false){
|
|
return Binary::readByte($this->get(1), $signed);
|
|
}
|
|
|
|
public function putByte($v){
|
|
$this->buffer .= Binary::writeByte($v);
|
|
}
|
|
|
|
public function getShort(){
|
|
return $this->endianness === self::BIG_ENDIAN ? Binary::readShort($this->get(2)) : Binary::readLShort($this->get(2));
|
|
}
|
|
|
|
public function putShort($v){
|
|
$this->buffer .= $this->endianness === self::BIG_ENDIAN ? Binary::writeShort($v) : Binary::writeLShort($v);
|
|
}
|
|
|
|
public function getInt(){
|
|
return $this->endianness === self::BIG_ENDIAN ? Binary::readInt($this->get(4)) : Binary::readLInt($this->get(4));
|
|
}
|
|
|
|
public function putInt($v){
|
|
$this->buffer .= $this->endianness === self::BIG_ENDIAN ? Binary::writeInt($v) : Binary::writeLInt($v);
|
|
}
|
|
|
|
public function getLong(){
|
|
return $this->endianness === self::BIG_ENDIAN ? Binary::readLong($this->get(8)) : Binary::readLLong($this->get(8));
|
|
}
|
|
|
|
public function putLong($v){
|
|
$this->buffer .= $this->endianness === self::BIG_ENDIAN ? Binary::writeLong($v) : Binary::writeLLong($v);
|
|
}
|
|
|
|
public function getFloat(){
|
|
return $this->endianness === self::BIG_ENDIAN ? Binary::readFloat($this->get(4)) : Binary::readLFloat($this->get(4));
|
|
}
|
|
|
|
public function putFloat($v){
|
|
$this->buffer .= $this->endianness === self::BIG_ENDIAN ? Binary::writeFloat($v) : Binary::writeLFloat($v);
|
|
}
|
|
|
|
public function getDouble(){
|
|
return $this->endianness === self::BIG_ENDIAN ? Binary::readDouble($this->get(8)) : Binary::readLDouble($this->get(8));
|
|
}
|
|
|
|
public function putDouble($v){
|
|
$this->buffer .= $this->endianness === self::BIG_ENDIAN ? Binary::writeDouble($v) : Binary::writeLDouble($v);
|
|
}
|
|
|
|
public function getString(){
|
|
return $this->get($this->getShort());
|
|
}
|
|
|
|
public function putString($v){
|
|
$this->putShort(strlen($v));
|
|
$this->buffer .= $v;
|
|
}
|
|
|
|
public function getData(){
|
|
return $this->data;
|
|
}
|
|
|
|
public function setData(Compound $data){
|
|
$this->data = $data;
|
|
}
|
|
|
|
} |