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;
|
2020-08-20 07:03:05 +00:00
|
|
|
|
using System.Collections.Concurrent;
|
2020-07-13 01:19:30 +00:00
|
|
|
|
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-09-14 16:20:34 +00:00
|
|
|
|
private static readonly ConcurrentDictionary<string, byte[]> Lookup = new ConcurrentDictionary<string, byte[]>();
|
2020-07-13 01:19:30 +00:00
|
|
|
|
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}";
|
|
|
|
|
|
2020-09-13 01:16:19 +00:00
|
|
|
|
public override int GetHashCode()
|
|
|
|
|
{
|
|
|
|
|
return Key.GetHashCode();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override bool Equals(object obj)
|
|
|
|
|
{
|
2021-02-18 19:15:08 +00:00
|
|
|
|
if (!(obj is SharedString other))
|
2020-09-13 01:16:19 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return Key.Equals(other.Key);
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 01:19:30 +00:00
|
|
|
|
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-08-20 07:03:05 +00:00
|
|
|
|
if (Lookup.ContainsKey(key))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Lookup.TryAdd(key, buffer);
|
2019-06-30 22:01:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private SharedString(byte[] buffer)
|
|
|
|
|
{
|
2020-08-20 07:03:05 +00:00
|
|
|
|
using (var 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)
|
|
|
|
|
{
|
2020-08-18 01:12:24 +00:00
|
|
|
|
return new SharedString(buffer ?? Array.Empty<byte>());
|
2019-06-30 22:01:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|