using System; using System.Buffers; using System.Buffers.Text; using System.Runtime.InteropServices; using System.Text; using Dalamud.Bootstrap.Crypto; namespace Dalamud.Bootstrap.SqexArg { internal sealed class EncodedArgument { private static char[] ChecksumTable = new char[] { 'f', 'X', '1', 'p', 'G', 't', 'd', 'S', '5', 'C', 'A', 'P', '4', '_', 'V', 'L' }; /// /// Denotes that no checksum is encoded. /// private const char NoChecksumMarker = '!'; private readonly ReadOnlyMemory m_data; public EncodedArgument(ReadOnlyMemory data, uint key) { m_data = data; m_key = key; } /// /// /// /// /// public static EncodedArgument Parse(string argument) { if (argument.Length <= 17) { // does not contain: //**sqex0003 + payload + checksum + **// var exMessage = $"The string ({argument}) is too short to parse encoded argument."; throw new SqexArgException(exMessage); } if (!argument.StartsWith("//**sqex0003") || !argument.EndsWith("**//")) { var exMessage = $"The string ({argument}) doesn't look like valid encoded argument format." + $"It either doesn't start with //**sqeex003 or end with **// marker."; throw new SqexArgException(exMessage); } var checksum = argument[^5]; var payload = DecodeUrlSafeBase64(argument.Substring(12, argument.Length - 1 - 12 - 4)); // //**sqex0003, checksum, **// // ... } /// /// /// /// /// /// private static Blowfish RecoverKey(ReadOnlySpan payload, char checksum) { var (keyFragment, step) = RecoverKeyFragmentFromChecksum(checksum); var keyCandicate = keyFragment; while (true) { // try with keyCandicate try { keyCandicate = checked(keyCandicate + step); } catch (OverflowException) { break; } } } /// /// /// /// /// /// private static (uint keyFragment, uint step) RecoverKeyFragmentFromChecksum(char checksum) { if (checksum == NoChecksumMarker) { return (0x0001_0000, 0x0001_0000); } return MemoryExtensions.IndexOf(ChecksumTable, checksum) switch { -1 => throw new SqexArgException($"{checksum} is not a valid checksum character."), var index => ((uint) (index << 16), 0x0010_0000) }; } /// /// Converts url safe variant of base64 string to bytes. /// /// A url-safe variant of base64 string. private static byte[] DecodeUrlSafeBase64(string payload) { var base64Str = payload .Replace('-', '+') .Replace('_', '/'); return Convert.FromBase64String(base64Str); } } }