*/
validate(hotp, count) {
const codeString = this.generate(count);
- return codeString === hotp.trim();
+ const codeStringB = Buffer.from(codeString);
+ const hotpB = Buffer.from(hotp.trim());
+ return codeStringB.length === hotpB.length
+ && crypto.timingSafeEqual(hotpB, codeStringB);
}
/**
'use strict';
const HOTP = require('./hotp');
+const crypto = require('node:crypto');
class TimeBasedOneTimePassword extends HOTP {
/**
*/
validate(hotp, count) {
const counter = count ?? this.counter;
+ const hotpB = Buffer.from(hotp.trim());
for (const offset of this.driftOffsets) {
const codeString = this.generate(counter + offset);
- if (codeString === hotp.trim()) {
+ const codeStringB = Buffer.from(codeString);
+ if (hotpB.length === codeString.length && crypto.timingSafeEqual(hotpB, codeStringB)) {
return true;
}
}