忘了在哪里看到的了,如果作者看到请谅解,
自用好长一段时间了,因为做爬虫用到,想起来了就分享出来
public class GoogleAuthenticator
{
///
/// 初始化验证码生成规则
///
/// 秘钥(手机使用Base32码),不可以为空
/// 验证码间隔多久刷新一次(默认30秒和google同步)
public GoogleAuthenticator(long duration = 30, string key = "SGOUNKTWDYUJ5RUX")
{
this.SERECT_KEY = key.ToLower();
// this.SERECT_KEY_MOBILE = Base32Encoding.ToString(Encoding.UTF8.GetBytes(key));
this.SERECT_KEY_MOBILE = key;
this.DURATION_TIME = duration;
}
///
/// 间隔时间
///
private long DURATION_TIME { get; set; }
///
/// 迭代次数
///
private long COUNTER
{
get
{
return (long)(DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc)).TotalSeconds / DURATION_TIME;
}
}
///
/// 秘钥
///
private string SERECT_KEY { get; set; }
///
/// 手机端输入的秘钥
///
private string SERECT_KEY_MOBILE { get; set; }
///
/// 到期秒数
///
public long EXPIRE_SECONDS
{
get
{
return (DURATION_TIME - (long)(DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc)).TotalSeconds % DURATION_TIME);
}
}
///
/// 获取手机端秘钥
///
///
public string GetMobilePhoneKey()
{
if (SERECT_KEY_MOBILE == null)
throw new ArgumentNullException("SERECT_KEY_MOBILE");
return SERECT_KEY_MOBILE;
}
///
/// 生成认证码
///
/// 返回验证码
public string GenerateCode()
{
return GenerateHashedCode(SERECT_KEY, COUNTER);
}
public string GetMobleCode()
{
return GenerateMobleHashedCode(SERECT_KEY_MOBILE);
}
///
/// 按照次数生成哈希编码,服务端
///
/// 秘钥
/// 迭代次数
/// 生成位数
/// 返回验证码
private string GenerateHashedCode(string secret, long iterationNumber, int digits = 6)
{
byte[] counter = BitConverter.GetBytes(iterationNumber);
if (BitConverter.IsLittleEndian)
Array.Reverse(counter);
byte[] key = Encoding.ASCII.GetBytes(secret);
HMACSHA1 hmac = new HMACSHA1(key, true);
byte[] hash = hmac.ComputeHash(counter);
int offset = hash[hash.Length - 1] & 0xf;
int binary =
((hash[offset] & 0x7f) << 24)
| ((hash[offset + 1] & 0xff) << 16)
| ((hash[offset + 2] & 0xff) << 8)
| (hash[offset + 3] & 0xff);
int password = binary % (int)Math.Pow(10, digits); // 6 digits
return password.ToString(new string('0', digits));
}
private string GenerateMobleHashedCode(string secret, int digits = 6)
{
byte[] counter = BitConverter.GetBytes(COUNTER);
if (BitConverter.IsLittleEndian)
Array.Reverse(counter);
byte[] key = Base32Encoding.ToBytes(secret);
HMACSHA1 hmac = new HMACSHA1(key, true);
byte[] hash = hmac.ComputeHash(counter);
int offset = hash[hash.Length - 1] & 0xf;
int binary =
((hash[offset] & 0x7f) << 24)
| ((hash[offset + 1] & 0xff) << 16)
| ((hash[offset + 2] & 0xff) << 8)
| (hash[offset + 3] & 0xff);
int password = binary % (int)Math.Pow(10, digits); // 6 digits
return password.ToString(new string('0', digits));
}
}
public static class Base32Encoding
{
public static byte[] ToBytes(string input)
{
if (string.IsNullOrEmpty(input))
{
throw new ArgumentNullException("input");
}
input = input.TrimEnd('=');
int byteCount = input.Length * 5 / 8;
byte[] returnArray = new byte[byteCount];
byte curByte = 0, bitsRemaining = 8;
int mask = 0, arrayIndex = 0;
foreach (char c in input)
{
int cValue = CharToValue(c);
if (bitsRemaining > 5)
{
mask = cValue << (bitsRemaining - 5);
curByte = (byte)(curByte | mask);
bitsRemaining -= 5;
}
else
{
mask = cValue >> (5 - bitsRemaining);
curByte = (byte)(curByte | mask);
returnArray[arrayIndex++] = curByte;
curByte = (byte)(cValue << (3 + bitsRemaining));
bitsRemaining += 3;
}
}
if (arrayIndex != byteCount)
{
returnArray[arrayIndex] = curByte;
}
return returnArray;
}
public static string ToString(byte[] input)
{
if (input == null || input.Length == 0)
{
throw new ArgumentNullException("input");
}
int charCount = (int)Math.Ceiling(input.Length / 5d) * 8;
char[] returnArray = new char[charCount];
byte nextChar = 0, bitsRemaining = 5;
int arrayIndex = 0;
foreach (byte b in input)
{
nextChar = (byte)(nextChar | (b >> (8 - bitsRemaining)));
returnArray[arrayIndex++] = ValueToChar(nextChar);
if (bitsRemaining < 4)
{
nextChar = (byte)((b >> (3 - bitsRemaining)) & 31);
returnArray[arrayIndex++] = ValueToChar(nextChar);
bitsRemaining += 5;
}
bitsRemaining -= 3;
nextChar = (byte)((b << bitsRemaining) & 31);
}
if (arrayIndex != charCount)
{
returnArray[arrayIndex++] = ValueToChar(nextChar);
while (arrayIndex != charCount) returnArray[arrayIndex++] = '=';
}
return new string(returnArray);
}
private static int CharToValue(char c)
{
var value = (int)c;
if (value < 91 && value > 64)
{
return value - 65;
}
if (value < 56 && value > 49)
{
return value - 24;
}
if (value < 123 && value > 96)
{
return value - 97;
}
throw new ArgumentException("Character is not a Base32 character.", "c");
}
private static char ValueToChar(byte b)
{
if (b < 26)
{
return (char)(b + 65);
}
if (b < 32)
{
return (char)(b + 24);
}
throw new ArgumentException("Byte is not a value Base32 value.", "b");
}
}
用法:
GoogleAuthenticator googleAuthenticator = new GoogleAuthenticator(30, "xxxxxxx");
var mobileKey = googleAuthenticator.GetMobilePhoneKey();
string gcode = googleAuthenticator.GetMobleCode();