Skip to content

Commit a4c6aab

Browse files
committed
Argument checking with better error messages
1 parent 58a55ad commit a4c6aab

File tree

2 files changed

+93
-98
lines changed

2 files changed

+93
-98
lines changed

libotp.ts

+82-87
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,75 @@ import * as url from 'url'
77

88
const debug = _debug('libotp')
99

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+
1079
function byteSizeForAlgo(algorithm: string): number {
1180
switch (algorithm) {
1281
case 'sha1':
@@ -16,7 +85,7 @@ function byteSizeForAlgo(algorithm: string): number {
1685
case 'sha512':
1786
return 64
1887
default:
19-
debug('Unrecognized hash algorithm `%s`', algorithm)
88+
throw new Error('Unrecognized hash algorithm `' + algorithm + '`')
2089
}
2190
}
2291

@@ -42,23 +111,6 @@ function padSecret(secret: Buffer, byteSize: number): Buffer {
42111
return secret
43112
}
44113

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-
62114
/**
63115
* Generate a base32-encoded random secret.
64116
*
@@ -182,8 +234,8 @@ abstract class OTP {
182234
*/
183235
constructor(params: BaseParams) {
184236
// 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')
187239

188240
// check secret
189241
this.secret = params.secret
@@ -196,36 +248,9 @@ abstract class OTP {
196248
}
197249
}
198250

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)
229254
if (params.label) this.label = params.label
230255
if (params.issuer) this.issuer = params.issuer
231256

@@ -402,9 +427,7 @@ abstract class OTP {
402427
const counter = this.counter
403428

404429
// required options
405-
if (!this.label) {
406-
throw new Error('missing label')
407-
}
430+
if (!this.label) throw new Error('Missing label value')
408431

409432
// convert secret to base32
410433
this._getSecret()
@@ -478,9 +501,7 @@ abstract class OTP {
478501
*/
479502
export class HOTP extends OTP {
480503
public readonly type: string = 'hotp'
481-
482-
protected _counter: number
483-
public get counter(): number { return this._counter }
504+
public counter: number
484505

485506
/**
486507
* Constructor.
@@ -504,12 +525,7 @@ export class HOTP extends OTP {
504525
*/
505526
constructor(params: HOTPParams) {
506527
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)
513529
}
514530

515531
/**
@@ -521,7 +537,7 @@ export class HOTP extends OTP {
521537
* @return {number} The counter value.
522538
*/
523539
public next(): number {
524-
return ++this._counter
540+
return ++this.counter
525541
}
526542

527543
/**
@@ -534,7 +550,7 @@ export class HOTP extends OTP {
534550
const delta = this.diff(token)
535551
const ok = delta !== false
536552
if (ok && delta > 0) {
537-
this._counter += <number>delta
553+
this.counter += <number>delta
538554
}
539555
return ok
540556
}
@@ -637,30 +653,9 @@ export class TOTP extends OTP {
637653
*/
638654
constructor(params: TOTPParams) {
639655
super(params)
640-
641-
// check time
642656
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)
664659
}
665660

666661
/**

test/url_test.js

+11-11
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ var url = require('url');
1616
new OTP({
1717
label: 'that'
1818
}).url();
19-
}, /missing secret/);
19+
}, /missing secret/i);
2020
});
2121

2222
it('should require label', function () {
@@ -25,7 +25,7 @@ var url = require('url');
2525
secret: 'hello',
2626
counter: 0
2727
}).url();
28-
}, /missing label/);
28+
}, /missing label/i);
2929
});
3030

3131
it('should validate algorithm', function () {
@@ -36,7 +36,7 @@ var url = require('url');
3636
algorithm: 'hello',
3737
counter: 0
3838
}).url();
39-
}, /invalid algorithm `hello`/);
39+
}, /invalid algorithm `hello`/i);
4040
assert.ok(new OTP({
4141
secret: 'hello',
4242
label: 'that',
@@ -65,7 +65,7 @@ var url = require('url');
6565
digits: 'hello',
6666
counter: 0
6767
}).url();
68-
}, /invalid digits/);
68+
}, /invalid digits/i);
6969
// Non-6 and non-8 digits should not throw, but should have a warn message
7070
assert.doesNotThrow(function () {
7171
new OTP({
@@ -74,15 +74,15 @@ var url = require('url');
7474
digits: 12,
7575
counter: 0
7676
}).url();
77-
}, /invalid digits/);
77+
}, /invalid digits/i);
7878
assert.throws(function () {
7979
new OTP({
8080
secret: 'hello',
8181
label: 'that',
8282
digits: '7',
8383
counter: 0
8484
}).url();
85-
}, /invalid digits/);
85+
}, /invalid digits/i);
8686
assert.ok(new OTP({
8787
secret: 'hello',
8888
label: 'that',
@@ -102,15 +102,15 @@ var url = require('url');
102102
digits: '6',
103103
counter: 0
104104
}).url();
105-
}, /invalid digits/);
105+
}, /invalid digits/i);
106106
assert.throws(function () {
107107
new OTP({
108108
secret: 'hello',
109109
label: 'that',
110110
digits: '8',
111111
counter: 0
112112
}).url();
113-
}, /invalid digits/);
113+
}, /invalid digits/i);
114114
});
115115

116116
if (klass == 'TOTP') {
@@ -122,7 +122,7 @@ var url = require('url');
122122
period: 'hello',
123123
counter: 0
124124
}).url();
125-
}, /invalid period/);
125+
}, /invalid period/i);
126126
assert.ok(new OTP({
127127
secret: 'hello',
128128
label: 'that',
@@ -142,15 +142,15 @@ var url = require('url');
142142
period: '60',
143143
counter: 0
144144
}).url();
145-
}, /invalid period/);
145+
}, /invalid period/i);
146146
assert.throws(function () {
147147
new OTP({
148148
secret: 'hello',
149149
label: 'that',
150150
period: '121',
151151
counter: 0
152152
}).url();
153-
}, /invalid period/);
153+
}, /invalid period/i);
154154
});
155155
}
156156

0 commit comments

Comments
 (0)