using System; using System.Collections.Generic; using System.Text; using System.IO; using System.IO.Compression; using Substrate.Core; namespace Substrate.Nbt { /// /// Contains the root node of an NBT tree and handles IO of tree nodes. /// /// /// NBT, or Named Byte Tag, is a tree-based data structure for storing most Minecraft data. /// NBT_Tree is more of a helper class for NBT trees that handles reading and writing nodes to data streams. /// Most of the API takes a TagValue or derived node as the root of the tree, rather than an NBT_Tree object itself. /// public class NbtTree : ICopyable { private Stream _stream = null; private TagNodeCompound _root = null; private string _rootName = ""; private static TagNodeNull _nulltag = new TagNodeNull(); /// /// Gets the root node of this tree. /// public TagNodeCompound Root { get { return _root; } } /// /// Gets or sets the name of the tree's root node. /// public string Name { get { return _rootName; } set { _rootName = value; } } /// /// Constructs a wrapper around a new NBT tree with an empty root node. /// public NbtTree () { _root = new TagNodeCompound(); } /// /// Constructs a wrapper around another NBT tree. /// /// The root node of an NBT tree. public NbtTree (TagNodeCompound tree) { _root = tree; } /// /// Constructs a wrapper around another NBT tree and gives it a name. /// /// The root node of an NBT tree. /// The name for the root node. public NbtTree (TagNodeCompound tree, string name) { _root = tree; _rootName = name; } /// /// Constructs and wrapper around a new NBT tree parsed from a source data stream. /// /// An open, readable data stream containing NBT data. public NbtTree (Stream s) { ReadFrom(s); } /// /// Rebuild the internal NBT tree from a source data stream. /// /// An open, readable data stream containing NBT data. public void ReadFrom (Stream s) { if (s != null) { _stream = s; _root = ReadRoot(); _stream = null; } } /// /// Writes out the internal NBT tree to a destination data stream. /// /// An open, writable data stream. public void WriteTo (Stream s) { if (s != null) { _stream = s; if (_root != null) { WriteTag(_rootName, _root); } _stream = null; } } private TagNode ReadValue (TagType type) { switch (type) { case TagType.TAG_END: return null; case TagType.TAG_BYTE: return ReadByte(); case TagType.TAG_SHORT: return ReadShort(); case TagType.TAG_INT: return ReadInt(); case TagType.TAG_LONG: return ReadLong(); case TagType.TAG_FLOAT: return ReadFloat(); case TagType.TAG_DOUBLE: return ReadDouble(); case TagType.TAG_BYTE_ARRAY: return ReadByteArray(); case TagType.TAG_STRING: return ReadString(); case TagType.TAG_LIST: return ReadList(); case TagType.TAG_COMPOUND: return ReadCompound(); case TagType.TAG_INT_ARRAY: return ReadIntArray(); } throw new Exception(); } private TagNode ReadByte () { int gzByte = _stream.ReadByte(); if (gzByte == -1) { throw new NBTException(NBTException.MSG_GZIP_ENDOFSTREAM); } TagNodeByte val = new TagNodeByte((byte)gzByte); return val; } private TagNode ReadShort () { byte[] gzBytes = new byte[2]; _stream.Read(gzBytes, 0, 2); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } TagNodeShort val = new TagNodeShort(BitConverter.ToInt16(gzBytes, 0)); return val; } private TagNode ReadInt () { byte[] gzBytes = new byte[4]; _stream.Read(gzBytes, 0, 4); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } TagNodeInt val = new TagNodeInt(BitConverter.ToInt32(gzBytes, 0)); return val; } private TagNode ReadLong () { byte[] gzBytes = new byte[8]; _stream.Read(gzBytes, 0, 8); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } TagNodeLong val = new TagNodeLong(BitConverter.ToInt64(gzBytes, 0)); return val; } private TagNode ReadFloat () { byte[] gzBytes = new byte[4]; _stream.Read(gzBytes, 0, 4); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } TagNodeFloat val = new TagNodeFloat(BitConverter.ToSingle(gzBytes, 0)); return val; } private TagNode ReadDouble () { byte[] gzBytes = new byte[8]; _stream.Read(gzBytes, 0, 8); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } TagNodeDouble val = new TagNodeDouble(BitConverter.ToDouble(gzBytes, 0)); return val; } private TagNode ReadByteArray () { byte[] lenBytes = new byte[4]; _stream.Read(lenBytes, 0, 4); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } int length = BitConverter.ToInt32(lenBytes, 0); if (length < 0) { throw new NBTException(NBTException.MSG_READ_NEG); } byte[] data = new byte[length]; _stream.Read(data, 0, length); TagNodeByteArray val = new TagNodeByteArray(data); return val; } private TagNode ReadString () { byte[] lenBytes = new byte[2]; _stream.Read(lenBytes, 0, 2); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } short len = BitConverter.ToInt16(lenBytes, 0); if (len < 0) { throw new NBTException(NBTException.MSG_READ_NEG); } byte[] strBytes = new byte[len]; _stream.Read(strBytes, 0, len); System.Text.Encoding str = Encoding.UTF8; TagNodeString val = new TagNodeString(str.GetString(strBytes)); return val; } private TagNode ReadList () { int gzByte = _stream.ReadByte(); if (gzByte == -1) { throw new NBTException(NBTException.MSG_GZIP_ENDOFSTREAM); } TagNodeList val = new TagNodeList((TagType)gzByte); if (val.ValueType > (TagType)Enum.GetValues(typeof(TagType)).GetUpperBound(0)) { throw new NBTException(NBTException.MSG_READ_TYPE); } byte[] lenBytes = new byte[4]; _stream.Read(lenBytes, 0, 4); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } int length = BitConverter.ToInt32(lenBytes, 0); if (length < 0) { throw new NBTException(NBTException.MSG_READ_NEG); } if (val.ValueType == TagType.TAG_END) return new TagNodeList(TagType.TAG_BYTE); for (int i = 0; i < length; i++) { val.Add(ReadValue(val.ValueType)); } return val; } private TagNode ReadCompound () { TagNodeCompound val = new TagNodeCompound(); while (ReadTag(val)) ; return val; } private TagNode ReadIntArray () { byte[] lenBytes = new byte[4]; _stream.Read(lenBytes, 0, 4); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } int length = BitConverter.ToInt32(lenBytes, 0); if (length < 0) { throw new NBTException(NBTException.MSG_READ_NEG); } int[] data = new int[length]; byte[] buffer = new byte[4]; for (int i = 0; i < length; i++) { _stream.Read(buffer, 0, 4); if (BitConverter.IsLittleEndian) { Array.Reverse(buffer); } data[i] = BitConverter.ToInt32(buffer, 0); } TagNodeIntArray val = new TagNodeIntArray(data); return val; } private TagNodeCompound ReadRoot () { TagType type = (TagType)_stream.ReadByte(); if (type == TagType.TAG_COMPOUND) { _rootName = ReadString().ToTagString().Data; // name return ReadValue(type) as TagNodeCompound; } return null; } private bool ReadTag (TagNodeCompound parent) { TagType type = (TagType)_stream.ReadByte(); if (type != TagType.TAG_END) { string name = ReadString().ToTagString().Data; parent[name] = ReadValue(type); return true; } return false; } private void WriteValue (TagNode val) { switch (val.GetTagType()) { case TagType.TAG_END: break; case TagType.TAG_BYTE: WriteByte(val.ToTagByte()); break; case TagType.TAG_SHORT: WriteShort(val.ToTagShort()); break; case TagType.TAG_INT: WriteInt(val.ToTagInt()); break; case TagType.TAG_LONG: WriteLong(val.ToTagLong()); break; case TagType.TAG_FLOAT: WriteFloat(val.ToTagFloat()); break; case TagType.TAG_DOUBLE: WriteDouble(val.ToTagDouble()); break; case TagType.TAG_BYTE_ARRAY: WriteByteArray(val.ToTagByteArray()); break; case TagType.TAG_STRING: WriteString(val.ToTagString()); break; case TagType.TAG_LIST: WriteList(val.ToTagList()); break; case TagType.TAG_COMPOUND: WriteCompound(val.ToTagCompound()); break; case TagType.TAG_INT_ARRAY: WriteIntArray(val.ToTagIntArray()); break; } } private void WriteByte (TagNodeByte val) { _stream.WriteByte(val.Data); } private void WriteShort (TagNodeShort val) { byte[] gzBytes = BitConverter.GetBytes(val.Data); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } _stream.Write(gzBytes, 0, 2); } private void WriteInt (TagNodeInt val) { byte[] gzBytes = BitConverter.GetBytes(val.Data); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } _stream.Write(gzBytes, 0, 4); } private void WriteLong (TagNodeLong val) { byte[] gzBytes = BitConverter.GetBytes(val.Data); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } _stream.Write(gzBytes, 0, 8); } private void WriteFloat (TagNodeFloat val) { byte[] gzBytes = BitConverter.GetBytes(val.Data); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } _stream.Write(gzBytes, 0, 4); } private void WriteDouble (TagNodeDouble val) { byte[] gzBytes = BitConverter.GetBytes(val.Data); if (BitConverter.IsLittleEndian) { Array.Reverse(gzBytes); } _stream.Write(gzBytes, 0, 8); } private void WriteByteArray (TagNodeByteArray val) { byte[] lenBytes = BitConverter.GetBytes(val.Length); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } _stream.Write(lenBytes, 0, 4); _stream.Write(val.Data, 0, val.Length); } private void WriteString (TagNodeString val) { System.Text.Encoding str = Encoding.UTF8; byte[] gzBytes = str.GetBytes(val.Data); byte[] lenBytes = BitConverter.GetBytes((short)gzBytes.Length); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } _stream.Write(lenBytes, 0, 2); _stream.Write(gzBytes, 0, gzBytes.Length); } private void WriteList (TagNodeList val) { byte[] lenBytes = BitConverter.GetBytes(val.Count); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } _stream.WriteByte((byte)val.ValueType); _stream.Write(lenBytes, 0, 4); foreach (TagNode v in val) { WriteValue(v); } } private void WriteCompound (TagNodeCompound val) { foreach (KeyValuePair item in val) { WriteTag(item.Key, item.Value); } WriteTag(null, _nulltag); } private void WriteIntArray (TagNodeIntArray val) { byte[] lenBytes = BitConverter.GetBytes(val.Length); if (BitConverter.IsLittleEndian) { Array.Reverse(lenBytes); } _stream.Write(lenBytes, 0, 4); byte[] data = new byte[val.Length * 4]; for (int i = 0; i < val.Length; i++) { byte[] buffer = BitConverter.GetBytes(val.Data[i]); if (BitConverter.IsLittleEndian) { Array.Reverse(buffer); } Array.Copy(buffer, 0, data, i * 4, 4); } _stream.Write(data, 0, data.Length); } private void WriteTag (string name, TagNode val) { _stream.WriteByte((byte)val.GetTagType()); if (val.GetTagType() != TagType.TAG_END) { WriteString(name); WriteValue(val); } } #region ICopyable Members /// /// Creates a deep copy of the NBT_Tree and underlying nodes. /// /// A new NBT_tree. public NbtTree Copy () { NbtTree tree = new NbtTree(); tree._root = _root.Copy() as TagNodeCompound; return tree; } #endregion } // TODO: Revise exceptions? public class NBTException : Exception { public const String MSG_GZIP_ENDOFSTREAM = "Gzip Error: Unexpected end of stream"; public const String MSG_READ_NEG = "Read Error: Negative length"; public const String MSG_READ_TYPE = "Read Error: Invalid value type"; public NBTException () { } public NBTException (String msg) : base(msg) { } public NBTException (String msg, Exception innerException) : base(msg, innerException) { } } public class InvalidNBTObjectException : Exception { } public class InvalidTagException : Exception { } public class InvalidValueException : Exception { } }