@@ -7,6 +7,75 @@ import * as url from 'url'
7
7
8
8
const debug = _debug ( 'libotp' )
9
9
10
+ function checkWindow ( window ) : number {
11
+ if ( Math . floor ( window ) !== window ) {
12
+ throw new Error ( 'Invalid window `' + window + '`' )
13
+ }
14
+ return window
15
+ }
16
+
17
+ function checkDigits ( digits ) : number {
18
+ if ( Math . floor ( digits ) !== digits ) {
19
+ throw new Error ( 'Invalid digits `' + digits + '`' )
20
+ } else if ( digits !== 6 ) {
21
+ debug ( 'Compatibility could be improved by using the default digits of 6.' )
22
+ }
23
+ return digits
24
+ }
25
+
26
+ function checkAlgorithm ( algorithm ) : string {
27
+ algorithm = algorithm . toLowerCase ( )
28
+ if ( algorithm !== 'sha1' ) {
29
+ debug ( 'Compatibility could be improved by using the default algorithm' +
30
+ ' of sha1.' )
31
+ }
32
+ return algorithm
33
+ }
34
+
35
+ function checkCounter ( counter ) : number {
36
+ if ( counter == null ) {
37
+ throw new Error ( 'Missing counter value' )
38
+ }
39
+ if ( Math . floor ( counter ) !== counter ) {
40
+ throw new Error ( 'Invalid counter `' + counter + '`' )
41
+ }
42
+ return counter
43
+ }
44
+
45
+ function checkTime ( time : Date | number | ( ( ) => Date | number ) ) : number | ( ( ) => number ) {
46
+ if ( typeof time === 'function' ) {
47
+ const fn : ( ( ) => any ) = time
48
+ time = fn ( )
49
+ if ( time instanceof Date ) {
50
+ return ( ) => Math . floor ( fn ( ) / 1000 )
51
+ } else if ( typeof time === 'number' ) {
52
+ return ( ) => Math . floor ( fn ( ) )
53
+ }
54
+ } else if ( time instanceof Date ) {
55
+ return + time / 1000
56
+ } else if ( typeof time === 'number' ) {
57
+ return Math . floor ( < number > time )
58
+ }
59
+ throw new Error ( 'invalid time ' + time )
60
+ }
61
+
62
+ function checkEpoch ( epoch ) : number {
63
+ if ( Math . floor ( epoch ) !== epoch ) {
64
+ throw new Error ( 'Invalid epoch `' + epoch + '`' )
65
+ }
66
+ return epoch
67
+ }
68
+
69
+ function checkPeriod ( period ) : number {
70
+ if ( Math . floor ( period ) !== period || period <= 0 ) {
71
+ throw new Error ( 'Invalid period `' + period + '`' )
72
+ } else if ( period !== 30 ) {
73
+ debug ( 'Compatibility could be improved by using the default period' +
74
+ ' of 30 seconds.' )
75
+ }
76
+ return period
77
+ }
78
+
10
79
function byteSizeForAlgo ( algorithm : string ) : number {
11
80
switch ( algorithm ) {
12
81
case 'sha1' :
@@ -16,7 +85,7 @@ function byteSizeForAlgo(algorithm: string): number {
16
85
case 'sha512' :
17
86
return 64
18
87
default :
19
- debug ( 'Unrecognized hash algorithm `%s`' , algorithm )
88
+ throw new Error ( 'Unrecognized hash algorithm `' + algorithm + '`' )
20
89
}
21
90
}
22
91
@@ -42,23 +111,6 @@ function padSecret(secret: Buffer, byteSize: number): Buffer {
42
111
return secret
43
112
}
44
113
45
- function checkTime ( time : Date | number | ( ( ) => Date | number ) ) : number | ( ( ) => number ) {
46
- if ( typeof time === 'function' ) {
47
- const fn : ( ( ) => any ) = time
48
- time = fn ( )
49
- if ( time instanceof Date ) {
50
- return ( ) => Math . floor ( fn ( ) / 1000 )
51
- } else if ( typeof time === 'number' ) {
52
- return ( ) => Math . floor ( fn ( ) )
53
- }
54
- } else if ( time instanceof Date ) {
55
- return + time / 1000
56
- } else if ( typeof time === 'number' ) {
57
- return Math . floor ( < number > time )
58
- }
59
- throw new Error ( 'invalid time ' + time )
60
- }
61
-
62
114
/**
63
115
* Generate a base32-encoded random secret.
64
116
*
@@ -182,8 +234,8 @@ abstract class OTP {
182
234
*/
183
235
constructor ( params : BaseParams ) {
184
236
// required parameters
185
- if ( ! params ) throw new Error ( 'missing params ' )
186
- if ( ! params . secret ) throw new Error ( 'missing secret' )
237
+ if ( ! params ) throw new Error ( 'Missing parameters ' )
238
+ if ( ! params . secret ) throw new Error ( 'Missing secret value ' )
187
239
188
240
// check secret
189
241
this . secret = params . secret
@@ -196,36 +248,9 @@ abstract class OTP {
196
248
}
197
249
}
198
250
199
- // check digits
200
- const digits = params . digits
201
- if ( digits ) {
202
- if ( Math . floor ( digits ) !== digits ) {
203
- throw new Error ( 'invalid digits' )
204
- } else if ( digits !== 6 ) {
205
- debug ( 'Compatibility could be improved by using the default' +
206
- ' digits of 6.' )
207
- }
208
- this . digits = digits
209
- }
210
-
211
- // check window
212
- const window = params . window
213
- if ( window ) {
214
- if ( Math . floor ( window ) !== window ) {
215
- throw new Error ( 'invalid window' )
216
- }
217
- this . window = window
218
- }
219
-
220
- if ( params . algorithm ) {
221
- const algorithm = params . algorithm . toLowerCase ( )
222
- if ( algorithm !== 'sha1' ) {
223
- debug ( 'Compatibility could be improved by using the default' +
224
- ' algorithm of sha1.' )
225
- }
226
- this . algorithm = algorithm
227
- }
228
-
251
+ if ( params . digits ) this . digits = checkDigits ( params . digits )
252
+ if ( params . window ) this . window = checkWindow ( params . window )
253
+ if ( params . algorithm ) this . algorithm = checkAlgorithm ( params . algorithm )
229
254
if ( params . label ) this . label = params . label
230
255
if ( params . issuer ) this . issuer = params . issuer
231
256
@@ -402,9 +427,7 @@ abstract class OTP {
402
427
const counter = this . counter
403
428
404
429
// required options
405
- if ( ! this . label ) {
406
- throw new Error ( 'missing label' )
407
- }
430
+ if ( ! this . label ) throw new Error ( 'Missing label value' )
408
431
409
432
// convert secret to base32
410
433
this . _getSecret ( )
@@ -478,9 +501,7 @@ abstract class OTP {
478
501
*/
479
502
export class HOTP extends OTP {
480
503
public readonly type : string = 'hotp'
481
-
482
- protected _counter : number
483
- public get counter ( ) : number { return this . _counter }
504
+ public counter : number
484
505
485
506
/**
486
507
* Constructor.
@@ -504,12 +525,7 @@ export class HOTP extends OTP {
504
525
*/
505
526
constructor ( params : HOTPParams ) {
506
527
super ( params )
507
-
508
- // check-assign counter
509
- const counter = params . counter
510
- if ( counter == null ) throw new Error ( 'missing counter' )
511
- if ( Math . floor ( counter ) !== counter ) throw new Error ( 'invalid counter' )
512
- this . _counter = counter
528
+ this . counter = checkCounter ( params . counter )
513
529
}
514
530
515
531
/**
@@ -521,7 +537,7 @@ export class HOTP extends OTP {
521
537
* @return {number } The counter value.
522
538
*/
523
539
public next ( ) : number {
524
- return ++ this . _counter
540
+ return ++ this . counter
525
541
}
526
542
527
543
/**
@@ -534,7 +550,7 @@ export class HOTP extends OTP {
534
550
const delta = this . diff ( token )
535
551
const ok = delta !== false
536
552
if ( ok && delta > 0 ) {
537
- this . _counter += < number > delta
553
+ this . counter += < number > delta
538
554
}
539
555
return ok
540
556
}
@@ -637,30 +653,9 @@ export class TOTP extends OTP {
637
653
*/
638
654
constructor ( params : TOTPParams ) {
639
655
super ( params )
640
-
641
- // check time
642
656
if ( params . time ) this . time = checkTime ( params . time )
643
-
644
- // check epoch
645
- if ( params . epoch ) {
646
- if ( Math . floor ( params . epoch ) !== params . epoch ) {
647
- throw new Error ( 'invalid epoch' )
648
- }
649
- this . epoch = params . epoch
650
- }
651
-
652
- // check period
653
- if ( params . period ) {
654
- if ( Math . floor ( params . period ) !== params . period ) {
655
- throw new Error ( 'invalid period' )
656
- } else if ( this . period <= 0 ) {
657
- throw new Error ( 'invalid period <= 0' )
658
- } else if ( this . period !== 30 ) {
659
- debug ( 'Compatibility could be improved by using the default period' +
660
- ' of 30 seconds.' )
661
- }
662
- this . period = params . period
663
- }
657
+ if ( params . epoch ) this . epoch = checkEpoch ( params . epoch )
658
+ if ( params . period ) this . period = checkPeriod ( params . period )
664
659
}
665
660
666
661
/**
0 commit comments