2019-01-26 00:39:37 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
2019-06-08 03:43:28 +00:00
|
|
|
|
using System.Linq;
|
2019-01-26 00:39:37 +00:00
|
|
|
|
using System.Text;
|
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
using RobloxFiles.BinaryFormat;
|
2019-02-01 17:19:20 +00:00
|
|
|
|
using RobloxFiles.BinaryFormat.Chunks;
|
2019-06-30 22:01:19 +00:00
|
|
|
|
using RobloxFiles.DataTypes;
|
2019-02-01 17:19:20 +00:00
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
namespace RobloxFiles
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-05-19 04:44:51 +00:00
|
|
|
|
public class BinaryRobloxFile : RobloxFile
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
// Header Specific
|
|
|
|
|
public const string MagicHeader = "<roblox!\x89\xff\x0d\x0a\x1a\x0a";
|
|
|
|
|
|
|
|
|
|
public ushort Version;
|
2019-06-30 22:01:19 +00:00
|
|
|
|
public uint NumClasses;
|
2019-01-29 09:50:55 +00:00
|
|
|
|
public uint NumInstances;
|
2019-06-08 03:43:28 +00:00
|
|
|
|
public long Reserved;
|
2019-01-29 09:50:55 +00:00
|
|
|
|
|
|
|
|
|
// Runtime Specific
|
2019-05-19 04:44:51 +00:00
|
|
|
|
public List<BinaryRobloxFileChunk> Chunks = new List<BinaryRobloxFileChunk>();
|
2019-01-29 09:50:55 +00:00
|
|
|
|
public override string ToString() => GetType().Name;
|
|
|
|
|
|
|
|
|
|
public Instance[] Instances;
|
2019-06-30 22:01:19 +00:00
|
|
|
|
public INST[] Classes;
|
2019-05-17 06:14:04 +00:00
|
|
|
|
|
2019-06-08 03:43:28 +00:00
|
|
|
|
internal META META = null;
|
|
|
|
|
internal SSTR SSTR = null;
|
|
|
|
|
|
|
|
|
|
public bool HasMetadata => (META != null);
|
|
|
|
|
public Dictionary<string, string> Metadata => META?.Data;
|
|
|
|
|
|
|
|
|
|
public bool HasSharedStrings => (SSTR != null);
|
2019-06-30 22:01:19 +00:00
|
|
|
|
public IReadOnlyDictionary<uint, SharedString> SharedStrings => SSTR?.Strings;
|
2019-05-19 04:44:51 +00:00
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
public BinaryRobloxFile()
|
2019-05-19 04:44:51 +00:00
|
|
|
|
{
|
|
|
|
|
Name = "BinaryRobloxFile";
|
|
|
|
|
ParentLocked = true;
|
2019-06-08 03:43:28 +00:00
|
|
|
|
Referent = "-1";
|
2019-05-19 04:44:51 +00:00
|
|
|
|
}
|
2019-01-29 09:50:55 +00:00
|
|
|
|
|
2019-05-19 04:44:51 +00:00
|
|
|
|
protected override void ReadFile(byte[] contents)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
|
|
|
|
using (MemoryStream file = new MemoryStream(contents))
|
2019-06-08 03:43:28 +00:00
|
|
|
|
using (BinaryRobloxFileReader reader = new BinaryRobloxFileReader(this, file))
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
// Verify the signature of the file.
|
2019-01-26 00:39:37 +00:00
|
|
|
|
byte[] binSignature = reader.ReadBytes(14);
|
|
|
|
|
string signature = Encoding.UTF7.GetString(binSignature);
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
if (signature != MagicHeader)
|
2019-01-30 06:36:56 +00:00
|
|
|
|
throw new InvalidDataException("Provided file's signature does not match BinaryRobloxFile.MagicHeader!");
|
2019-01-26 00:39:37 +00:00
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
// Read header data.
|
2019-01-26 00:39:37 +00:00
|
|
|
|
Version = reader.ReadUInt16();
|
2019-06-30 22:01:19 +00:00
|
|
|
|
NumClasses = reader.ReadUInt32();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
NumInstances = reader.ReadUInt32();
|
2019-06-08 03:43:28 +00:00
|
|
|
|
Reserved = reader.ReadInt64();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
|
|
|
|
|
// Begin reading the file chunks.
|
|
|
|
|
bool reading = true;
|
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
Classes = new INST[NumClasses];
|
2019-01-29 09:50:55 +00:00
|
|
|
|
Instances = new Instance[NumInstances];
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
2019-01-26 00:39:37 +00:00
|
|
|
|
while (reading)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2019-05-19 04:44:51 +00:00
|
|
|
|
BinaryRobloxFileChunk chunk = new BinaryRobloxFileChunk(reader);
|
2019-06-08 03:43:28 +00:00
|
|
|
|
string chunkType = chunk.ChunkType;
|
2019-01-26 00:39:37 +00:00
|
|
|
|
|
2019-06-08 03:43:28 +00:00
|
|
|
|
IBinaryFileChunk handler = null;
|
|
|
|
|
|
|
|
|
|
switch (chunkType)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
|
|
|
|
case "INST":
|
2019-06-08 03:43:28 +00:00
|
|
|
|
handler = new INST();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
break;
|
|
|
|
|
case "PROP":
|
2019-06-08 03:43:28 +00:00
|
|
|
|
handler = new PROP();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
break;
|
|
|
|
|
case "PRNT":
|
2019-06-08 03:43:28 +00:00
|
|
|
|
handler = new PRNT();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
break;
|
|
|
|
|
case "META":
|
2019-06-08 03:43:28 +00:00
|
|
|
|
handler = new META();
|
2019-05-17 06:14:04 +00:00
|
|
|
|
break;
|
|
|
|
|
case "SSTR":
|
2019-06-08 03:43:28 +00:00
|
|
|
|
handler = new SSTR();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
break;
|
|
|
|
|
case "END\0":
|
2019-06-30 22:01:19 +00:00
|
|
|
|
Chunks.Add(chunk);
|
2019-01-26 00:39:37 +00:00
|
|
|
|
reading = false;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2019-06-08 03:43:28 +00:00
|
|
|
|
Console.WriteLine("BinaryRobloxFile - Unhandled chunk-type: {0}!", chunkType);
|
2019-01-26 00:39:37 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
|
|
|
|
if (handler != null)
|
|
|
|
|
{
|
|
|
|
|
using (BinaryRobloxFileReader dataReader = chunk.GetDataReader(this))
|
|
|
|
|
handler.LoadFromReader(dataReader);
|
|
|
|
|
|
|
|
|
|
chunk.Handler = handler;
|
|
|
|
|
Chunks.Add(chunk);
|
|
|
|
|
}
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
catch (EndOfStreamException)
|
|
|
|
|
{
|
|
|
|
|
throw new Exception("Unexpected end of file!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-05-17 12:08:06 +00:00
|
|
|
|
|
2019-05-19 04:44:51 +00:00
|
|
|
|
public override void Save(Stream stream)
|
2019-05-17 12:08:06 +00:00
|
|
|
|
{
|
2019-06-08 03:43:28 +00:00
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Generate the chunk data.
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
using (var writer = new BinaryRobloxFileWriter(this))
|
|
|
|
|
{
|
|
|
|
|
// Clear the existing data.
|
|
|
|
|
Referent = "-1";
|
|
|
|
|
Chunks.Clear();
|
|
|
|
|
|
|
|
|
|
NumInstances = 0;
|
2019-06-30 22:01:19 +00:00
|
|
|
|
NumClasses = 0;
|
2019-06-11 01:27:57 +00:00
|
|
|
|
SSTR = null;
|
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
// Recursively capture all instances and classes.
|
2019-06-08 03:43:28 +00:00
|
|
|
|
writer.RecordInstances(Children);
|
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
// Apply the recorded instances and classes.
|
|
|
|
|
writer.ApplyClassMap();
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
|
|
|
|
// Write the INST chunks.
|
2019-06-30 22:01:19 +00:00
|
|
|
|
foreach (INST inst in Classes)
|
|
|
|
|
writer.SaveChunk(inst);
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
|
|
|
|
// Write the PROP chunks.
|
2019-06-30 22:01:19 +00:00
|
|
|
|
foreach (INST inst in Classes)
|
2019-06-08 03:43:28 +00:00
|
|
|
|
{
|
2019-06-30 22:01:19 +00:00
|
|
|
|
Dictionary<string, PROP> props = PROP.CollectProperties(writer, inst);
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
|
|
|
|
foreach (string propName in props.Keys)
|
|
|
|
|
{
|
|
|
|
|
PROP prop = props[propName];
|
2019-06-30 22:01:19 +00:00
|
|
|
|
writer.SaveChunk(prop);
|
2019-06-08 03:43:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write the PRNT chunk.
|
|
|
|
|
PRNT parents = new PRNT();
|
2019-06-30 22:01:19 +00:00
|
|
|
|
writer.SaveChunk(parents);
|
|
|
|
|
|
2019-06-08 03:43:28 +00:00
|
|
|
|
// Write the SSTR chunk.
|
|
|
|
|
if (HasSharedStrings)
|
|
|
|
|
{
|
|
|
|
|
var sharedStrings = SSTR.SaveAsChunk(writer);
|
|
|
|
|
Chunks.Insert(0, sharedStrings);
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-11 01:27:57 +00:00
|
|
|
|
// Write the META chunk.
|
|
|
|
|
if (HasMetadata)
|
|
|
|
|
{
|
|
|
|
|
var metaChunk = META.SaveAsChunk(writer);
|
|
|
|
|
Chunks.Insert(0, metaChunk);
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-08 03:43:28 +00:00
|
|
|
|
// Write the END_ chunk.
|
2019-06-30 22:01:19 +00:00
|
|
|
|
var endChunk = writer.WriteEndChunk();
|
2019-06-08 03:43:28 +00:00
|
|
|
|
Chunks.Add(endChunk);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2019-06-30 22:01:19 +00:00
|
|
|
|
// Write the chunk buffers with the header data
|
2019-06-08 03:43:28 +00:00
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
using (BinaryWriter writer = new BinaryWriter(stream))
|
|
|
|
|
{
|
|
|
|
|
stream.Position = 0;
|
|
|
|
|
stream.SetLength(0);
|
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
writer.Write(MagicHeader
|
2019-06-08 03:43:28 +00:00
|
|
|
|
.Select(ch => (byte)ch)
|
2019-06-30 22:01:19 +00:00
|
|
|
|
.ToArray());
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
|
|
|
|
writer.Write(Version);
|
2019-06-30 22:01:19 +00:00
|
|
|
|
writer.Write(NumClasses);
|
2019-06-08 03:43:28 +00:00
|
|
|
|
writer.Write(NumInstances);
|
2019-06-30 22:01:19 +00:00
|
|
|
|
writer.Write(Reserved);
|
2019-06-08 03:43:28 +00:00
|
|
|
|
|
|
|
|
|
foreach (BinaryRobloxFileChunk chunk in Chunks)
|
|
|
|
|
{
|
|
|
|
|
if (chunk.HasWriteBuffer)
|
|
|
|
|
{
|
|
|
|
|
byte[] writeBuffer = chunk.WriteBuffer;
|
|
|
|
|
writer.Write(writeBuffer);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-05-17 12:08:06 +00:00
|
|
|
|
}
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
2019-06-08 05:18:00 +00:00
|
|
|
|
}
|