1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace TwoStepsAuthenticator
{
/// <summary>
/// Implementation of rfc6238 Time-Based One-Time Password Algorithm
/// </summary>
public class TimeAuthenticator : Authenticator
{
private static readonly Lazy<IUsedCodesManager> DefaultUsedCodeManager = new Lazy<IUsedCodesManager>(() => new UsedCodesManager());
private readonly Func<DateTime> NowFunc;
private readonly IUsedCodesManager UsedCodeManager;
private readonly int IntervalSeconds;
public TimeAuthenticator(IUsedCodesManager usedCodeManager = null, Func<DateTime> nowFunc = null, int intervalSeconds = 30)
{
this.NowFunc = (nowFunc == null) ? () => DateTime.Now : nowFunc;
this.UsedCodeManager = (usedCodeManager == null) ? DefaultUsedCodeManager.Value : usedCodeManager;
this.IntervalSeconds = intervalSeconds;
}
/// <summary>
/// Generates One-Time Password.
/// </summary>
/// <param name="secret">Shared Secret</param>
/// <returns>OTP</returns>
public string GetCode(string secret)
{
return GetCode(secret, NowFunc());
}
/// <summary>
/// Generates One-Time Password.
/// </summary>
/// <param name="secret">Shared Secret</param>
/// <param name="date">Time to use as challenge</param>
/// <returns>OTP</returns>
public string GetCode(string secret, DateTime date)
{
return GetCodeInternal(secret, (ulong)GetInterval(date));
}
/// <summary>
/// Checks if the passed code is valid.
/// </summary>
/// <param name="secret">Shared Secret</param>
/// <param name="code">OTP</param>
/// <param name="user">The user</param>
/// <returns>true if code matches</returns>
public bool CheckCode(string secret, string code, object user)
{
DateTime successfulTime = DateTime.MinValue;
return CheckCode(secret, code, user, out successfulTime);
}
/// <summary>
/// Checks if the passed code is valid.
/// </summary>
/// <param name="secret">Shared Secret</param>
/// <param name="code">OTP</param>
/// <param name="user">The user</param>
/// <param name="usedDateTime">Matching time if successful</param>
/// <returns>true if code matches</returns>
public bool CheckCode(string secret, string code, object user, out DateTime usedDateTime)
{
var baseTime = NowFunc();
DateTime successfulTime = DateTime.MinValue;
// We need to do this in constant time
var codeMatch = false;
for (int i = -2; i <= 1; i++)
{
var checkTime = baseTime.AddSeconds(IntervalSeconds * i);
var checkInterval = GetInterval(checkTime);
if (ConstantTimeEquals(GetCode(secret, checkTime), code) && (user == null || !UsedCodeManager.IsCodeUsed(checkInterval, code, user)))
{
codeMatch = true;
successfulTime = checkTime;
UsedCodeManager.AddCode(checkInterval, code, user);
}
}
usedDateTime = successfulTime;
return codeMatch;
}
/// <summary>
/// Checks if the passed code is valid.
/// </summary>
/// <param name="secret">Shared Secret</param>
/// <param name="code">OTP</param>
/// <returns>true if code matches</returns>
[Obsolete("The CheckCode method should only be used with a user object")]
public bool CheckCode(string secret, string code)
{
DateTime successfulTime = DateTime.MinValue;
return CheckCode(secret, code, null, out successfulTime);
}
private long GetInterval(DateTime dateTime)
{
TimeSpan ts = (dateTime.ToUniversalTime() - new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc));
return (long)ts.TotalSeconds / IntervalSeconds;
}
}
}
|