2019-06-30 22:01:19 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Text;
|
2020-07-13 01:19:30 +00:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using Konscious.Security.Cryptography;
|
2019-06-30 22:01:19 +00:00
|
|
|
|
|
|
|
|
|
namespace RobloxFiles.DataTypes
|
|
|
|
|
{
|
2020-07-13 01:19:30 +00:00
|
|
|
|
// SharedString is a datatype that takes a sequence of bytes and stores it in a
|
|
|
|
|
// lookup table that is shared by the entire file. It originally used MD5 for the
|
|
|
|
|
// hashing, but Roblox now uses Blake2B to avoid the obvious problems with using MD5.
|
|
|
|
|
|
|
|
|
|
// In practice the value of a SharedString does not have to match the hash of the
|
|
|
|
|
// data it represents, it just needs to be distinct and MUST be 16 bytes long.
|
|
|
|
|
// The XML format still uses 'md5' as its attribute key to the lookup table.
|
|
|
|
|
|
2019-06-30 22:01:19 +00:00
|
|
|
|
public class SharedString
|
|
|
|
|
{
|
2020-07-13 01:19:30 +00:00
|
|
|
|
private static Dictionary<string, byte[]> Lookup = new Dictionary<string, byte[]>();
|
|
|
|
|
public string Key { get; internal set; }
|
|
|
|
|
public string ComputedKey { get; internal set; }
|
2019-06-30 22:01:19 +00:00
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
public byte[] SharedValue => Find(ComputedKey ?? Key);
|
|
|
|
|
public override string ToString() => $"Key: {ComputedKey ?? Key}";
|
|
|
|
|
|
|
|
|
|
internal SharedString(string key)
|
|
|
|
|
{
|
|
|
|
|
Key = key;
|
|
|
|
|
}
|
2019-06-30 22:01:19 +00:00
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
internal static void Register(string key, byte[] buffer)
|
2019-06-30 22:01:19 +00:00
|
|
|
|
{
|
2020-07-13 01:19:30 +00:00
|
|
|
|
Lookup.Add(key, buffer);
|
2019-06-30 22:01:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private SharedString(byte[] buffer)
|
|
|
|
|
{
|
2020-07-13 01:19:30 +00:00
|
|
|
|
using (HMACBlake2B blake2B = new HMACBlake2B(16 * 8))
|
2019-06-30 22:01:19 +00:00
|
|
|
|
{
|
2020-07-13 01:19:30 +00:00
|
|
|
|
byte[] hash = blake2B.ComputeHash(buffer);
|
|
|
|
|
ComputedKey = Convert.ToBase64String(hash);
|
|
|
|
|
Key = ComputedKey;
|
2019-06-30 22:01:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
if (Lookup.ContainsKey(ComputedKey))
|
2019-06-30 22:01:19 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
Register(ComputedKey, buffer);
|
2019-06-30 22:01:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
public static byte[] Find(string key)
|
2019-06-30 22:01:19 +00:00
|
|
|
|
{
|
|
|
|
|
byte[] result = null;
|
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
if (Lookup.ContainsKey(key))
|
|
|
|
|
result = Lookup[key];
|
2019-06-30 22:01:19 +00:00
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static SharedString FromBuffer(byte[] buffer)
|
|
|
|
|
{
|
|
|
|
|
return new SharedString(buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static SharedString FromString(string value)
|
|
|
|
|
{
|
|
|
|
|
byte[] buffer = Encoding.UTF8.GetBytes(value);
|
|
|
|
|
return new SharedString(buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static SharedString FromBase64(string base64)
|
|
|
|
|
{
|
|
|
|
|
byte[] buffer = Convert.FromBase64String(base64);
|
|
|
|
|
return new SharedString(buffer);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|