2019-04-17 23:01:07 +03:00
|
|
|
|
using System;
|
|
|
|
|
|
|
|
|
|
namespace Bit.Core.Utilities
|
|
|
|
|
{
|
|
|
|
|
// ref: https://github.com/aspnet/Identity/blob/dev/src/Microsoft.Extensions.Identity.Core/Base32.cs
|
|
|
|
|
// with some modifications for cleaning input
|
|
|
|
|
public static class Base32
|
|
|
|
|
{
|
|
|
|
|
private static readonly string _base32Chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZ234567";
|
|
|
|
|
|
|
|
|
|
public static byte[] FromBase32(string input)
|
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (input == null)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
throw new ArgumentNullException(nameof(input));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
input = input.ToUpperInvariant();
|
|
|
|
|
var cleanedInput = string.Empty;
|
2020-03-28 16:16:28 +03:00
|
|
|
|
foreach (var c in input)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (_base32Chars.IndexOf(c) < 0)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cleanedInput += c;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
input = cleanedInput;
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (input.Length == 0)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
return new byte[0];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var output = new byte[input.Length * 5 / 8];
|
|
|
|
|
var bitIndex = 0;
|
|
|
|
|
var inputIndex = 0;
|
|
|
|
|
var outputBits = 0;
|
|
|
|
|
var outputIndex = 0;
|
|
|
|
|
|
2020-03-28 16:16:28 +03:00
|
|
|
|
while (outputIndex < output.Length)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
var byteIndex = _base32Chars.IndexOf(input[inputIndex]);
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (byteIndex < 0)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
throw new FormatException();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var bits = Math.Min(5 - bitIndex, 8 - outputBits);
|
|
|
|
|
output[outputIndex] <<= bits;
|
|
|
|
|
output[outputIndex] |= (byte)(byteIndex >> (5 - (bitIndex + bits)));
|
|
|
|
|
|
|
|
|
|
bitIndex += bits;
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (bitIndex >= 5)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
inputIndex++;
|
|
|
|
|
bitIndex = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
outputBits += bits;
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (outputBits >= 8)
|
2019-04-17 23:01:07 +03:00
|
|
|
|
{
|
|
|
|
|
outputIndex++;
|
|
|
|
|
outputBits = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return output;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|