TimestampNonceManager.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. using System.Buffers.Binary;
  2. using System.Collections.Concurrent;
  3. using System.Security.Cryptography;
  4. namespace PCC.App.Security;
  5. public class TimestampNonceManager : IDisposable
  6. {
  7. public static int TimestampLength => sizeof(long);
  8. private readonly ConcurrentDictionary<string, DateTime> _holds = new();
  9. private readonly TimeSpan _expiration;
  10. private readonly TimeSpan _maxTimeSkew;
  11. private readonly Timer _cleanupTimer;
  12. private readonly int _randomLength;
  13. public int NonceLength { get; }
  14. private bool _disposed;
  15. public TimestampNonceManager(int randomLength, TimeSpan expiration, TimeSpan maxTimeSkew)
  16. {
  17. NonceLength = TimestampLength + randomLength;
  18. _randomLength = randomLength;
  19. _expiration = expiration;
  20. _maxTimeSkew = maxTimeSkew;
  21. _cleanupTimer = new Timer(CleanupTimerCallback, null, expiration, expiration);
  22. }
  23. /// <summary> 生成一个新的nonce </summary>
  24. public (ReadOnlyMemory<byte> nonce, DateTimeOffset timestamp) NewNonce()
  25. {
  26. ThrowIfDisposed();
  27. var nonce = new byte[NonceLength];
  28. // 把时间戳放在nonce开头
  29. var timestamp = DateTimeOffset.UtcNow;
  30. BinaryPrimitives.TryWriteInt64LittleEndian(nonce, timestamp.ToUnixTimeMilliseconds());
  31. // 剩下的字节作为随机 nonce
  32. RandomNumberGenerator.Fill(nonce.AsSpan(TimestampLength));
  33. return (new ReadOnlyMemory<byte>(nonce), timestamp);
  34. }
  35. /// <summary> 生成一个新的nonce并追加payload </summary>
  36. public (ReadOnlyMemory<byte> payloadAndNonce, DateTimeOffset timestamp) NewNonceWithPayload(ReadOnlyMemory<byte> payload)
  37. {
  38. ThrowIfDisposed();
  39. var payloadAndNonce = new byte[NonceLength + payload.Length];
  40. // 把时间戳放在nonce开头
  41. var timestamp = DateTimeOffset.UtcNow;
  42. BinaryPrimitives.TryWriteInt64LittleEndian(payloadAndNonce, timestamp.ToUnixTimeMilliseconds());
  43. // 剩下的字节作为随机 nonce
  44. RandomNumberGenerator.Fill(payloadAndNonce.AsSpan(TimestampLength, _randomLength));
  45. // 追加 payload
  46. payload.Span.CopyTo(payloadAndNonce.AsSpan(NonceLength));
  47. return (new ReadOnlyMemory<byte>(payloadAndNonce), timestamp);
  48. }
  49. /// <summary> 验证nonce: null时间误差过大,false检测到重放攻击,true没问题 </summary>
  50. public bool? CheckValid(ReadOnlyMemory<byte> timestampNonce, out DateTimeOffset timestamp)
  51. {
  52. ThrowIfDisposed();
  53. if (timestampNonce.Length != NonceLength) throw new ArgumentException("invalid length", nameof(timestampNonce));
  54. timestamp = DateTimeOffset.FromUnixTimeMilliseconds(BinaryPrimitives.ReadInt64LittleEndian(timestampNonce.Span[..TimestampLength]));
  55. // 检查时间戳是否在允许的时间窗口内,超过最大时间差则拒绝
  56. if (Math.Abs((DateTimeOffset.UtcNow - timestamp).TotalMilliseconds) > _maxTimeSkew.TotalMilliseconds) return null;
  57. // 剩下的部分为真正的随机 nonce
  58. var nonce = Convert.ToHexString(timestampNonce[TimestampLength..].Span);
  59. // 如果 nonce 已存在,则为重放攻击,返回 false;否则添加并返回 true
  60. return _holds.TryAdd(nonce, DateTime.UtcNow);
  61. }
  62. /// <summary> 验证nonce并提取payload: null时间误差过大,false检测到重放攻击,true没问题 </summary>
  63. public (bool?, DateTimeOffset timestamp, ReadOnlyMemory<byte> payload) CheckValidAndExtractPayload(ReadOnlyMemory<byte> payloadAndNonce)
  64. {
  65. ThrowIfDisposed();
  66. if (payloadAndNonce.Length < NonceLength) throw new ArgumentException("invalid length", nameof(payloadAndNonce));
  67. //提取时间戳、nonce、payload
  68. var timestamp = DateTimeOffset.FromUnixTimeMilliseconds(BinaryPrimitives.ReadInt64LittleEndian(payloadAndNonce.Span[..TimestampLength]));
  69. var nonce = Convert.ToHexString(payloadAndNonce.Slice(TimestampLength, _randomLength).Span);
  70. var payload = payloadAndNonce[NonceLength..];
  71. // 检查时间戳是否在允许的时间窗口内, 超过最大时间差则拒绝
  72. if (Math.Abs((DateTimeOffset.UtcNow - timestamp).TotalMilliseconds) > _maxTimeSkew.TotalMilliseconds) return (null, timestamp, payload);
  73. // 如果 nonce 已存在,则为重放攻击,返回 false;否则添加并返回 true
  74. return (_holds.TryAdd(nonce, DateTime.UtcNow), timestamp, payload);
  75. }
  76. private void CleanupTimerCallback(object? state)
  77. {
  78. var now = DateTime.UtcNow;
  79. foreach (var item in _holds.Where(p => now - p.Value >= _expiration)) _holds.TryRemove(item.Key, out _);
  80. }
  81. private void ThrowIfDisposed() => ObjectDisposedException.ThrowIf(_disposed, this);
  82. public void Dispose()
  83. {
  84. if (_disposed) return;
  85. _cleanupTimer.Dispose();
  86. _disposed = true;
  87. GC.SuppressFinalize(this);
  88. }
  89. }