2020-05-30 08:19:47 +02:00
|
|
|
import { Logger } from 'util/logger';
|
|
|
|
import { YubiKey } from 'comp/app/yubikey';
|
|
|
|
|
|
|
|
const logger = new Logger('chal-resp');
|
|
|
|
|
|
|
|
const ChalRespCalculator = {
|
|
|
|
cache: {},
|
|
|
|
|
|
|
|
build(params) {
|
|
|
|
if (!params) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return challenge => {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const ts = logger.ts();
|
|
|
|
|
|
|
|
challenge = Buffer.from(challenge);
|
|
|
|
const hexChallenge = challenge.toString('hex');
|
|
|
|
|
2020-05-30 10:29:02 +02:00
|
|
|
const deviceCacheKey = `${params.vid}:${params.pid}:${params.serial}`;
|
|
|
|
const respFromCache = this.cache[deviceCacheKey]?.[hexChallenge];
|
|
|
|
if (respFromCache) {
|
2020-05-30 08:19:47 +02:00
|
|
|
logger.debug('Found ChalResp in cache');
|
2020-05-30 10:29:02 +02:00
|
|
|
return resolve(Buffer.from(respFromCache, 'hex'));
|
2020-05-30 08:19:47 +02:00
|
|
|
}
|
|
|
|
|
2020-05-30 10:29:02 +02:00
|
|
|
logger.debug('Calculating ChalResp using a YubiKey', params);
|
2020-05-30 08:19:47 +02:00
|
|
|
|
|
|
|
YubiKey.calculateChalResp(params, challenge, (err, response) => {
|
|
|
|
if (err) {
|
2020-05-30 10:29:02 +02:00
|
|
|
if (err.noKey) {
|
|
|
|
logger.debug('No YubiKey');
|
|
|
|
// TODO
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (err.touchRequested) {
|
|
|
|
logger.debug('YubiKey touch requested');
|
|
|
|
// TODO
|
|
|
|
return;
|
|
|
|
}
|
2020-05-30 08:19:47 +02:00
|
|
|
return reject(err);
|
|
|
|
}
|
2020-05-30 10:29:02 +02:00
|
|
|
|
|
|
|
if (!this.cache[deviceCacheKey]) {
|
|
|
|
this.cache[deviceCacheKey] = {};
|
|
|
|
}
|
|
|
|
this.cache[deviceCacheKey][hexChallenge] = response.toString('hex');
|
|
|
|
|
2020-05-30 08:19:47 +02:00
|
|
|
logger.info('Calculated ChalResp', logger.ts(ts));
|
|
|
|
resolve(response);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export { ChalRespCalculator };
|