blob: f4553ab8081e14f725029dd64c17e1c957dc1e73 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
/**
* @fileoverview Implements an enroll handler using USB gnubbies.
*/
'use strict';
/**
* @param {!EnrollHelperRequest} request The enroll request.
* @constructor
* @implements {RequestHandler}
*/
function UsbEnrollHandler(request) {
/** @private {!EnrollHelperRequest} */
this.request_ = request;
/** @private {Array<Gnubby>} */
this.waitingForTouchGnubbies_ = [];
/** @private {boolean} */
this.closed_ = false;
/** @private {boolean} */
this.notified_ = false;
}
/**
* Default timeout value in case the caller never provides a valid timeout.
* @const
*/
UsbEnrollHandler.DEFAULT_TIMEOUT_MILLIS = 30 * 1000;
/**
* @param {RequestHandlerCallback} cb Called back with the result of the
* request, and an optional source for the result.
* @return {boolean} Whether this handler could be run.
*/
UsbEnrollHandler.prototype.run = function(cb) {
var timeoutMillis = this.request_.timeoutSeconds ?
this.request_.timeoutSeconds * 1000 :
UsbEnrollHandler.DEFAULT_TIMEOUT_MILLIS;
/** @private {Countdown} */
this.timer_ =
DEVICE_FACTORY_REGISTRY.getCountdownFactory().createTimer(timeoutMillis);
this.enrollChallenges = this.request_.enrollChallenges;
/** @private {RequestHandlerCallback} */
this.cb_ = cb;
this.signer_ = new MultipleGnubbySigner(
true /* forEnroll */, this.signerCompleted_.bind(this),
this.signerFoundGnubby_.bind(this), timeoutMillis,
this.request_.logMsgUrl);
return this.signer_.doSign(this.request_.signData);
};
/** Closes this helper. */
UsbEnrollHandler.prototype.close = function() {
this.closed_ = true;
for (var i = 0; i < this.waitingForTouchGnubbies_.length; i++) {
this.waitingForTouchGnubbies_[i].closeWhenIdle();
}
this.waitingForTouchGnubbies_ = [];
if (this.signer_) {
this.signer_.close();
this.signer_ = null;
}
};
/**
* Called when a MultipleGnubbySigner completes its sign request.
* @param {boolean} anyPending Whether any gnubbies are pending.
* @private
*/
UsbEnrollHandler.prototype.signerCompleted_ = function(anyPending) {
if (!this.anyGnubbiesFound_ || this.anyTimeout_ || anyPending ||
this.timer_.expired()) {
this.notifyError_(DeviceStatusCodes.TIMEOUT_STATUS);
} else {
// Do nothing: signerFoundGnubby will have been called with each succeeding
// gnubby.
}
};
/**
* Called when a MultipleGnubbySigner finds a gnubby that can enroll.
* @param {MultipleSignerResult} signResult Signature results
* @param {boolean} moreExpected Whether the signer expects to report
* results from more gnubbies.
* @private
*/
UsbEnrollHandler.prototype.signerFoundGnubby_ = function(
signResult, moreExpected) {
if (!signResult.code) {
// If the signer reports a gnubby can sign, report this immediately to the
// caller, as the gnubby is already enrolled. Map ok to WRONG_DATA, so the
// caller knows what to do.
this.notifyError_(DeviceStatusCodes.WRONG_DATA_STATUS);
} else if (SingleGnubbySigner.signErrorIndicatesInvalidKeyHandle(
signResult.code)) {
var gnubby = signResult['gnubby'];
// A valid helper request contains at least one enroll challenge, so use
// the app id hash from the first challenge.
var appIdHash = this.request_.enrollChallenges[0].appIdHash;
DEVICE_FACTORY_REGISTRY.getGnubbyFactory().notEnrolledPrerequisiteCheck(
gnubby, appIdHash, this.gnubbyPrerequisitesChecked_.bind(this));
} else {
// Unexpected error in signing? Send this immediately to the caller.
this.notifyError_(signResult.code);
}
};
/**
* Called with the result of a gnubby prerequisite check.
* @param {number} rc The result of the prerequisite check.
* @param {Gnubby=} opt_gnubby The gnubby whose prerequisites were checked.
* @private
*/
UsbEnrollHandler.prototype.gnubbyPrerequisitesChecked_ = function(
rc, opt_gnubby) {
if (rc || this.timer_.expired()) {
// Do nothing:
// If the timer is expired, the signerCompleted_ callback will indicate
// timeout to the caller.
// If there's an error, this gnubby is ineligible, but there's nothing we
// can do about that here.
return;
}
// If the callback succeeded, the gnubby is not null.
var gnubby = /** @type {Gnubby} */ (opt_gnubby);
this.anyGnubbiesFound_ = true;
this.waitingForTouchGnubbies_.push(gnubby);
this.matchEnrollVersionToGnubby_(gnubby);
};
/**
* Attempts to match the gnubby's U2F version with an appropriate enroll
* challenge.
* @param {Gnubby} gnubby Gnubby instance
* @private
*/
UsbEnrollHandler.prototype.matchEnrollVersionToGnubby_ = function(gnubby) {
if (!gnubby) {
console.warn(UTIL_fmt('no gnubby, WTF?'));
return;
}
gnubby.version(this.gnubbyVersioned_.bind(this, gnubby));
};
/**
* Called with the result of a version command.
* @param {Gnubby} gnubby Gnubby instance
* @param {number} rc result of version command.
* @param {ArrayBuffer=} data version.
* @private
*/
UsbEnrollHandler.prototype.gnubbyVersioned_ = function(gnubby, rc, data) {
if (rc) {
this.removeWrongVersionGnubby_(gnubby);
return;
}
var version = UTIL_BytesToString(new Uint8Array(data || null));
this.tryEnroll_(gnubby, version);
};
/**
* Drops the gnubby from the list of eligible gnubbies.
* @param {Gnubby} gnubby Gnubby instance
* @private
*/
UsbEnrollHandler.prototype.removeWaitingGnubby_ = function(gnubby) {
gnubby.closeWhenIdle();
var index = this.waitingForTouchGnubbies_.indexOf(gnubby);
if (index >= 0) {
this.waitingForTouchGnubbies_.splice(index, 1);
}
};
/**
* Drops the gnubby from the list of eligible gnubbies, as it has the wrong
* version.
* @param {Gnubby} gnubby Gnubby instance
* @private
*/
UsbEnrollHandler.prototype.removeWrongVersionGnubby_ = function(gnubby) {
this.removeWaitingGnubby_(gnubby);
if (!this.waitingForTouchGnubbies_.length) {
// Whoops, this was the last gnubby.
this.anyGnubbiesFound_ = false;
if (this.timer_.expired()) {
this.notifyError_(DeviceStatusCodes.TIMEOUT_STATUS);
} else if (this.signer_) {
this.signer_.reScanDevices();
}
}
};
/**
* Attempts enrolling a particular gnubby with a challenge of the appropriate
* version.
* @param {Gnubby} gnubby Gnubby instance
* @param {string} version Protocol version
* @private
*/
UsbEnrollHandler.prototype.tryEnroll_ = function(gnubby, version) {
var challenge = this.getChallengeOfVersion_(version);
if (!challenge) {
this.removeWrongVersionGnubby_(gnubby);
return;
}
var appIdHashBase64 = challenge['appIdHash'];
if (DEVICE_FACTORY_REGISTRY.getIndividualAttestation()
.requestIndividualAttestation(appIdHashBase64)) {
this.tryEnrollComplete_(gnubby, version, true);
return;
}
if (!chrome.cryptotokenPrivate) {
this.tryEnrollComplete_(gnubby, version, false);
return;
}
chrome.cryptotokenPrivate.isAppIdHashInEnterpriseContext(
decodeWebSafeBase64ToArray(appIdHashBase64),
this.tryEnrollComplete_.bind(this, gnubby, version));
};
/**
* Attempts enrolling a particular gnubby with a challenge of the appropriate
* version.
* @param {Gnubby} gnubby Gnubby instance
* @param {string} version Protocol version
* @param {boolean} individualAttest whether to send the individual-attestation
* signal to the token.
* @private
*/
UsbEnrollHandler.prototype.tryEnrollComplete_ = function(
gnubby, version, individualAttest) {
var challenge = this.getChallengeOfVersion_(version);
var challengeValue = B64_decode(challenge['challengeHash']);
gnubby.enroll(
challengeValue, B64_decode(challenge['appIdHash']),
this.enrollCallback_.bind(this, gnubby, version), individualAttest);
};
/**
* Finds the (first) challenge of the given version in this helper's challenges.
* @param {string} version Protocol version
* @return {Object} challenge, if found, or null if not.
* @private
*/
UsbEnrollHandler.prototype.getChallengeOfVersion_ = function(version) {
for (var i = 0; i < this.enrollChallenges.length; i++) {
if (this.enrollChallenges[i]['version'] == version) {
return this.enrollChallenges[i];
}
}
return null;
};
/**
* Called with the result of an enroll request to a gnubby.
* @param {Gnubby} gnubby Gnubby instance
* @param {string} version Protocol version
* @param {number} code Status code
* @param {ArrayBuffer=} infoArray Returned data
* @private
*/
UsbEnrollHandler.prototype.enrollCallback_ = function(
gnubby, version, code, infoArray) {
if (this.notified_) {
// Enroll completed after previous success or failure. Disregard.
return;
}
switch (code) {
case -GnubbyDevice.GONE:
// Close this gnubby.
this.removeWaitingGnubby_(gnubby);
if (!this.waitingForTouchGnubbies_.length) {
// Last enroll attempt is complete and last gnubby is gone.
this.anyGnubbiesFound_ = false;
if (this.timer_.expired()) {
this.notifyError_(DeviceStatusCodes.TIMEOUT_STATUS);
} else if (this.signer_) {
this.signer_.reScanDevices();
}
}
break;
case DeviceStatusCodes.WAIT_TOUCH_STATUS:
case DeviceStatusCodes.BUSY_STATUS:
case DeviceStatusCodes.TIMEOUT_STATUS:
if (this.timer_.expired()) {
// Record that at least one gnubby timed out, to return a timeout status
// from the complete callback if no other eligible gnubbies are found.
/** @private {boolean} */
this.anyTimeout_ = true;
// Close this gnubby.
this.removeWaitingGnubby_(gnubby);
if (!this.waitingForTouchGnubbies_.length) {
// Last enroll attempt is complete: return this error.
console.log(
UTIL_fmt('timeout (' + code.toString(16) + ') enrolling'));
this.notifyError_(DeviceStatusCodes.TIMEOUT_STATUS);
}
} else {
DEVICE_FACTORY_REGISTRY.getCountdownFactory().createTimer(
UsbEnrollHandler.ENUMERATE_DELAY_INTERVAL_MILLIS,
this.tryEnroll_.bind(this, gnubby, version));
}
break;
case DeviceStatusCodes.OK_STATUS:
var appIdHash = this.request_.enrollChallenges[0].appIdHash;
DEVICE_FACTORY_REGISTRY.getGnubbyFactory().postEnrollAction(
gnubby, appIdHash, (rc) => {
if (rc == DeviceStatusCodes.OK_STATUS) {
var info = B64_encode(new Uint8Array(infoArray || []));
this.notifySuccess_(version, info);
} else {
this.notifyError_(rc);
}
});
break;
default:
console.log(UTIL_fmt('Failed to enroll gnubby: ' + code));
this.notifyError_(code);
break;
}
};
/**
* How long to delay between repeated enroll attempts, in milliseconds.
* @const
*/
UsbEnrollHandler.ENUMERATE_DELAY_INTERVAL_MILLIS = 200;
/**
* Notifies the callback with an error code.
* @param {number} code The error code to report.
* @private
*/
UsbEnrollHandler.prototype.notifyError_ = function(code) {
if (this.notified_ || this.closed_) {
return;
}
this.notified_ = true;
this.close();
var reply = {'type': 'enroll_helper_reply', 'code': code};
this.cb_(reply);
};
/**
* @param {string} version Protocol version
* @param {string} info B64 encoded success data
* @private
*/
UsbEnrollHandler.prototype.notifySuccess_ = function(version, info) {
if (this.notified_ || this.closed_) {
return;
}
this.notified_ = true;
this.close();
var reply = {
'type': 'enroll_helper_reply',
'code': DeviceStatusCodes.OK_STATUS,
'version': version,
'enrollData': info
};
this.cb_(reply);
};