u2f-api.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749
  1. //Copyright 2014-2015 Google Inc. All rights reserved.
  2. //Use of this source code is governed by a BSD-style
  3. //license that can be found in the LICENSE file or at
  4. //https://developers.google.com/open-source/licenses/bsd
  5. /**
  6. * @fileoverview The U2F api.
  7. */
  8. 'use strict';
  9. /**
  10. * Namespace for the U2F api.
  11. * @type {Object}
  12. */
  13. var u2f = u2f || {};
  14. /**
  15. * FIDO U2F Javascript API Version
  16. * @number
  17. */
  18. var js_api_version;
  19. /**
  20. * The U2F extension id
  21. * @const {string}
  22. */
  23. // The Chrome packaged app extension ID.
  24. // Uncomment this if you want to deploy a server instance that uses
  25. // the package Chrome app and does not require installing the U2F Chrome extension.
  26. u2f.EXTENSION_ID = 'kmendfapggjehodndflmmgagdbamhnfd';
  27. // The U2F Chrome extension ID.
  28. // Uncomment this if you want to deploy a server instance that uses
  29. // the U2F Chrome extension to authenticate.
  30. // u2f.EXTENSION_ID = 'pfboblefjcgdjicmnffhdgionmgcdmne';
  31. /**
  32. * Message types for messsages to/from the extension
  33. * @const
  34. * @enum {string}
  35. */
  36. u2f.MessageTypes = {
  37. 'U2F_REGISTER_REQUEST': 'u2f_register_request',
  38. 'U2F_REGISTER_RESPONSE': 'u2f_register_response',
  39. 'U2F_SIGN_REQUEST': 'u2f_sign_request',
  40. 'U2F_SIGN_RESPONSE': 'u2f_sign_response',
  41. 'U2F_GET_API_VERSION_REQUEST': 'u2f_get_api_version_request',
  42. 'U2F_GET_API_VERSION_RESPONSE': 'u2f_get_api_version_response'
  43. };
  44. /**
  45. * Response status codes
  46. * @const
  47. * @enum {number}
  48. */
  49. u2f.ErrorCodes = {
  50. 'OK': 0,
  51. 'OTHER_ERROR': 1,
  52. 'BAD_REQUEST': 2,
  53. 'CONFIGURATION_UNSUPPORTED': 3,
  54. 'DEVICE_INELIGIBLE': 4,
  55. 'TIMEOUT': 5
  56. };
  57. /**
  58. * A message for registration requests
  59. * @typedef {{
  60. * type: u2f.MessageTypes,
  61. * appId: ?string,
  62. * timeoutSeconds: ?number,
  63. * requestId: ?number
  64. * }}
  65. */
  66. u2f.U2fRequest;
  67. /**
  68. * A message for registration responses
  69. * @typedef {{
  70. * type: u2f.MessageTypes,
  71. * responseData: (u2f.Error | u2f.RegisterResponse | u2f.SignResponse),
  72. * requestId: ?number
  73. * }}
  74. */
  75. u2f.U2fResponse;
  76. /**
  77. * An error object for responses
  78. * @typedef {{
  79. * errorCode: u2f.ErrorCodes,
  80. * errorMessage: ?string
  81. * }}
  82. */
  83. u2f.Error;
  84. /**
  85. * Data object for a single sign request.
  86. * @typedef {enum {BLUETOOTH_RADIO, BLUETOOTH_LOW_ENERGY, USB, NFC}}
  87. */
  88. u2f.Transport;
  89. /**
  90. * Data object for a single sign request.
  91. * @typedef {Array<u2f.Transport>}
  92. */
  93. u2f.Transports;
  94. /**
  95. * Data object for a single sign request.
  96. * @typedef {{
  97. * version: string,
  98. * challenge: string,
  99. * keyHandle: string,
  100. * appId: string
  101. * }}
  102. */
  103. u2f.SignRequest;
  104. /**
  105. * Data object for a sign response.
  106. * @typedef {{
  107. * keyHandle: string,
  108. * signatureData: string,
  109. * clientData: string
  110. * }}
  111. */
  112. u2f.SignResponse;
  113. /**
  114. * Data object for a registration request.
  115. * @typedef {{
  116. * version: string,
  117. * challenge: string
  118. * }}
  119. */
  120. u2f.RegisterRequest;
  121. /**
  122. * Data object for a registration response.
  123. * @typedef {{
  124. * version: string,
  125. * keyHandle: string,
  126. * transports: Transports,
  127. * appId: string
  128. * }}
  129. */
  130. u2f.RegisterResponse;
  131. /**
  132. * Data object for a registered key.
  133. * @typedef {{
  134. * version: string,
  135. * keyHandle: string,
  136. * transports: ?Transports,
  137. * appId: ?string
  138. * }}
  139. */
  140. u2f.RegisteredKey;
  141. /**
  142. * Data object for a get API register response.
  143. * @typedef {{
  144. * js_api_version: number
  145. * }}
  146. */
  147. u2f.GetJsApiVersionResponse;
  148. //Low level MessagePort API support
  149. /**
  150. * Sets up a MessagePort to the U2F extension using the
  151. * available mechanisms.
  152. * @param {function((MessagePort|u2f.WrappedChromeRuntimePort_))} callback
  153. */
  154. u2f.getMessagePort = function(callback) {
  155. if (typeof chrome != 'undefined' && chrome.runtime) {
  156. // The actual message here does not matter, but we need to get a reply
  157. // for the callback to run. Thus, send an empty signature request
  158. // in order to get a failure response.
  159. var msg = {
  160. type: u2f.MessageTypes.U2F_SIGN_REQUEST,
  161. signRequests: []
  162. };
  163. chrome.runtime.sendMessage(u2f.EXTENSION_ID, msg, function() {
  164. if (!chrome.runtime.lastError) {
  165. // We are on a whitelisted origin and can talk directly
  166. // with the extension.
  167. u2f.getChromeRuntimePort_(callback);
  168. } else {
  169. // chrome.runtime was available, but we couldn't message
  170. // the extension directly, use iframe
  171. u2f.getIframePort_(callback);
  172. }
  173. });
  174. } else if (u2f.isAndroidChrome_()) {
  175. u2f.getAuthenticatorPort_(callback);
  176. } else if (u2f.isIosChrome_()) {
  177. u2f.getIosPort_(callback);
  178. } else {
  179. // chrome.runtime was not available at all, which is normal
  180. // when this origin doesn't have access to any extensions.
  181. u2f.getIframePort_(callback);
  182. }
  183. };
  184. /**
  185. * Detect chrome running on android based on the browser's useragent.
  186. * @private
  187. */
  188. u2f.isAndroidChrome_ = function() {
  189. var userAgent = navigator.userAgent;
  190. return userAgent.indexOf('Chrome') != -1 &&
  191. userAgent.indexOf('Android') != -1;
  192. };
  193. /**
  194. * Detect chrome running on iOS based on the browser's platform.
  195. * @private
  196. */
  197. u2f.isIosChrome_ = function() {
  198. return ["iPhone", "iPad", "iPod"].indexOf(navigator.platform) > -1;
  199. };
  200. /**
  201. * Connects directly to the extension via chrome.runtime.connect.
  202. * @param {function(u2f.WrappedChromeRuntimePort_)} callback
  203. * @private
  204. */
  205. u2f.getChromeRuntimePort_ = function(callback) {
  206. var port = chrome.runtime.connect(u2f.EXTENSION_ID,
  207. {'includeTlsChannelId': true});
  208. setTimeout(function() {
  209. callback(new u2f.WrappedChromeRuntimePort_(port));
  210. }, 0);
  211. };
  212. /**
  213. * Return a 'port' abstraction to the Authenticator app.
  214. * @param {function(u2f.WrappedAuthenticatorPort_)} callback
  215. * @private
  216. */
  217. u2f.getAuthenticatorPort_ = function(callback) {
  218. setTimeout(function() {
  219. callback(new u2f.WrappedAuthenticatorPort_());
  220. }, 0);
  221. };
  222. /**
  223. * Return a 'port' abstraction to the iOS client app.
  224. * @param {function(u2f.WrappedIosPort_)} callback
  225. * @private
  226. */
  227. u2f.getIosPort_ = function(callback) {
  228. setTimeout(function() {
  229. callback(new u2f.WrappedIosPort_());
  230. }, 0);
  231. };
  232. /**
  233. * A wrapper for chrome.runtime.Port that is compatible with MessagePort.
  234. * @param {Port} port
  235. * @constructor
  236. * @private
  237. */
  238. u2f.WrappedChromeRuntimePort_ = function(port) {
  239. this.port_ = port;
  240. };
  241. /**
  242. * Format and return a sign request compliant with the JS API version supported by the extension.
  243. * @param {Array<u2f.SignRequest>} signRequests
  244. * @param {number} timeoutSeconds
  245. * @param {number} reqId
  246. * @return {Object}
  247. */
  248. u2f.formatSignRequest_ =
  249. function(appId, challenge, registeredKeys, timeoutSeconds, reqId) {
  250. if (js_api_version === undefined || js_api_version < 1.1) {
  251. // Adapt request to the 1.0 JS API
  252. var signRequests = [];
  253. for (var i = 0; i < registeredKeys.length; i++) {
  254. signRequests[i] = {
  255. version: registeredKeys[i].version,
  256. challenge: challenge,
  257. keyHandle: registeredKeys[i].keyHandle,
  258. appId: appId
  259. };
  260. }
  261. return {
  262. type: u2f.MessageTypes.U2F_SIGN_REQUEST,
  263. signRequests: signRequests,
  264. timeoutSeconds: timeoutSeconds,
  265. requestId: reqId
  266. };
  267. }
  268. // JS 1.1 API
  269. return {
  270. type: u2f.MessageTypes.U2F_SIGN_REQUEST,
  271. appId: appId,
  272. challenge: challenge,
  273. registeredKeys: registeredKeys,
  274. timeoutSeconds: timeoutSeconds,
  275. requestId: reqId
  276. };
  277. };
  278. /**
  279. * Format and return a register request compliant with the JS API version supported by the extension..
  280. * @param {Array<u2f.SignRequest>} signRequests
  281. * @param {Array<u2f.RegisterRequest>} signRequests
  282. * @param {number} timeoutSeconds
  283. * @param {number} reqId
  284. * @return {Object}
  285. */
  286. u2f.formatRegisterRequest_ =
  287. function(appId, registeredKeys, registerRequests, timeoutSeconds, reqId) {
  288. if (js_api_version === undefined || js_api_version < 1.1) {
  289. // Adapt request to the 1.0 JS API
  290. for (var i = 0; i < registerRequests.length; i++) {
  291. registerRequests[i].appId = appId;
  292. }
  293. var signRequests = [];
  294. for (var i = 0; i < registeredKeys.length; i++) {
  295. signRequests[i] = {
  296. version: registeredKeys[i].version,
  297. challenge: registerRequests[0],
  298. keyHandle: registeredKeys[i].keyHandle,
  299. appId: appId
  300. };
  301. }
  302. return {
  303. type: u2f.MessageTypes.U2F_REGISTER_REQUEST,
  304. signRequests: signRequests,
  305. registerRequests: registerRequests,
  306. timeoutSeconds: timeoutSeconds,
  307. requestId: reqId
  308. };
  309. }
  310. // JS 1.1 API
  311. return {
  312. type: u2f.MessageTypes.U2F_REGISTER_REQUEST,
  313. appId: appId,
  314. registerRequests: registerRequests,
  315. registeredKeys: registeredKeys,
  316. timeoutSeconds: timeoutSeconds,
  317. requestId: reqId
  318. };
  319. };
  320. /**
  321. * Posts a message on the underlying channel.
  322. * @param {Object} message
  323. */
  324. u2f.WrappedChromeRuntimePort_.prototype.postMessage = function(message) {
  325. this.port_.postMessage(message);
  326. };
  327. /**
  328. * Emulates the HTML 5 addEventListener interface. Works only for the
  329. * onmessage event, which is hooked up to the chrome.runtime.Port.onMessage.
  330. * @param {string} eventName
  331. * @param {function({data: Object})} handler
  332. */
  333. u2f.WrappedChromeRuntimePort_.prototype.addEventListener =
  334. function(eventName, handler) {
  335. var name = eventName.toLowerCase();
  336. if (name == 'message' || name == 'onmessage') {
  337. this.port_.onMessage.addListener(function(message) {
  338. // Emulate a minimal MessageEvent object
  339. handler({'data': message});
  340. });
  341. } else {
  342. console.error('WrappedChromeRuntimePort only supports onMessage');
  343. }
  344. };
  345. /**
  346. * Wrap the Authenticator app with a MessagePort interface.
  347. * @constructor
  348. * @private
  349. */
  350. u2f.WrappedAuthenticatorPort_ = function() {
  351. this.requestId_ = -1;
  352. this.requestObject_ = null;
  353. }
  354. /**
  355. * Launch the Authenticator intent.
  356. * @param {Object} message
  357. */
  358. u2f.WrappedAuthenticatorPort_.prototype.postMessage = function(message) {
  359. var intentUrl =
  360. u2f.WrappedAuthenticatorPort_.INTENT_URL_BASE_ +
  361. ';S.request=' + encodeURIComponent(JSON.stringify(message)) +
  362. ';end';
  363. document.location = intentUrl;
  364. };
  365. /**
  366. * Tells what type of port this is.
  367. * @return {String} port type
  368. */
  369. u2f.WrappedAuthenticatorPort_.prototype.getPortType = function() {
  370. return "WrappedAuthenticatorPort_";
  371. };
  372. /**
  373. * Emulates the HTML 5 addEventListener interface.
  374. * @param {string} eventName
  375. * @param {function({data: Object})} handler
  376. */
  377. u2f.WrappedAuthenticatorPort_.prototype.addEventListener = function(eventName, handler) {
  378. var name = eventName.toLowerCase();
  379. if (name == 'message') {
  380. var self = this;
  381. /* Register a callback to that executes when
  382. * chrome injects the response. */
  383. window.addEventListener(
  384. 'message', self.onRequestUpdate_.bind(self, handler), false);
  385. } else {
  386. console.error('WrappedAuthenticatorPort only supports message');
  387. }
  388. };
  389. /**
  390. * Callback invoked when a response is received from the Authenticator.
  391. * @param function({data: Object}) callback
  392. * @param {Object} message message Object
  393. */
  394. u2f.WrappedAuthenticatorPort_.prototype.onRequestUpdate_ =
  395. function(callback, message) {
  396. var messageObject = JSON.parse(message.data);
  397. var intentUrl = messageObject['intentURL'];
  398. var errorCode = messageObject['errorCode'];
  399. var responseObject = null;
  400. if (messageObject.hasOwnProperty('data')) {
  401. responseObject = /** @type {Object} */ (
  402. JSON.parse(messageObject['data']));
  403. }
  404. callback({'data': responseObject});
  405. };
  406. /**
  407. * Base URL for intents to Authenticator.
  408. * @const
  409. * @private
  410. */
  411. u2f.WrappedAuthenticatorPort_.INTENT_URL_BASE_ =
  412. 'intent:#Intent;action=com.google.android.apps.authenticator.AUTHENTICATE';
  413. /**
  414. * Wrap the iOS client app with a MessagePort interface.
  415. * @constructor
  416. * @private
  417. */
  418. u2f.WrappedIosPort_ = function() {};
  419. /**
  420. * Launch the iOS client app request
  421. * @param {Object} message
  422. */
  423. u2f.WrappedIosPort_.prototype.postMessage = function(message) {
  424. var str = JSON.stringify(message);
  425. var url = "u2f://auth?" + encodeURI(str);
  426. location.replace(url);
  427. };
  428. /**
  429. * Tells what type of port this is.
  430. * @return {String} port type
  431. */
  432. u2f.WrappedIosPort_.prototype.getPortType = function() {
  433. return "WrappedIosPort_";
  434. };
  435. /**
  436. * Emulates the HTML 5 addEventListener interface.
  437. * @param {string} eventName
  438. * @param {function({data: Object})} handler
  439. */
  440. u2f.WrappedIosPort_.prototype.addEventListener = function(eventName, handler) {
  441. var name = eventName.toLowerCase();
  442. if (name !== 'message') {
  443. console.error('WrappedIosPort only supports message');
  444. }
  445. };
  446. /**
  447. * Sets up an embedded trampoline iframe, sourced from the extension.
  448. * @param {function(MessagePort)} callback
  449. * @private
  450. */
  451. u2f.getIframePort_ = function(callback) {
  452. // Create the iframe
  453. var iframeOrigin = 'chrome-extension://' + u2f.EXTENSION_ID;
  454. var iframe = document.createElement('iframe');
  455. iframe.src = iframeOrigin + '/u2f-comms.html';
  456. iframe.setAttribute('style', 'display:none');
  457. document.body.appendChild(iframe);
  458. var channel = new MessageChannel();
  459. var ready = function(message) {
  460. if (message.data == 'ready') {
  461. channel.port1.removeEventListener('message', ready);
  462. callback(channel.port1);
  463. } else {
  464. console.error('First event on iframe port was not "ready"');
  465. }
  466. };
  467. channel.port1.addEventListener('message', ready);
  468. channel.port1.start();
  469. iframe.addEventListener('load', function() {
  470. // Deliver the port to the iframe and initialize
  471. iframe.contentWindow.postMessage('init', iframeOrigin, [channel.port2]);
  472. });
  473. };
  474. //High-level JS API
  475. /**
  476. * Default extension response timeout in seconds.
  477. * @const
  478. */
  479. u2f.EXTENSION_TIMEOUT_SEC = 30;
  480. /**
  481. * A singleton instance for a MessagePort to the extension.
  482. * @type {MessagePort|u2f.WrappedChromeRuntimePort_}
  483. * @private
  484. */
  485. u2f.port_ = null;
  486. /**
  487. * Callbacks waiting for a port
  488. * @type {Array<function((MessagePort|u2f.WrappedChromeRuntimePort_))>}
  489. * @private
  490. */
  491. u2f.waitingForPort_ = [];
  492. /**
  493. * A counter for requestIds.
  494. * @type {number}
  495. * @private
  496. */
  497. u2f.reqCounter_ = 0;
  498. /**
  499. * A map from requestIds to client callbacks
  500. * @type {Object.<number,(function((u2f.Error|u2f.RegisterResponse))
  501. * |function((u2f.Error|u2f.SignResponse)))>}
  502. * @private
  503. */
  504. u2f.callbackMap_ = {};
  505. /**
  506. * Creates or retrieves the MessagePort singleton to use.
  507. * @param {function((MessagePort|u2f.WrappedChromeRuntimePort_))} callback
  508. * @private
  509. */
  510. u2f.getPortSingleton_ = function(callback) {
  511. if (u2f.port_) {
  512. callback(u2f.port_);
  513. } else {
  514. if (u2f.waitingForPort_.length == 0) {
  515. u2f.getMessagePort(function(port) {
  516. u2f.port_ = port;
  517. u2f.port_.addEventListener('message',
  518. /** @type {function(Event)} */ (u2f.responseHandler_));
  519. // Careful, here be async callbacks. Maybe.
  520. while (u2f.waitingForPort_.length)
  521. u2f.waitingForPort_.shift()(u2f.port_);
  522. });
  523. }
  524. u2f.waitingForPort_.push(callback);
  525. }
  526. };
  527. /**
  528. * Handles response messages from the extension.
  529. * @param {MessageEvent.<u2f.Response>} message
  530. * @private
  531. */
  532. u2f.responseHandler_ = function(message) {
  533. var response = message.data;
  534. var reqId = response['requestId'];
  535. if (!reqId || !u2f.callbackMap_[reqId]) {
  536. console.error('Unknown or missing requestId in response.');
  537. return;
  538. }
  539. var cb = u2f.callbackMap_[reqId];
  540. delete u2f.callbackMap_[reqId];
  541. cb(response['responseData']);
  542. };
  543. /**
  544. * Dispatches an array of sign requests to available U2F tokens.
  545. * If the JS API version supported by the extension is unknown, it first sends a
  546. * message to the extension to find out the supported API version and then it sends
  547. * the sign request.
  548. * @param {string=} appId
  549. * @param {string=} challenge
  550. * @param {Array<u2f.RegisteredKey>} registeredKeys
  551. * @param {function((u2f.Error|u2f.SignResponse))} callback
  552. * @param {number=} opt_timeoutSeconds
  553. */
  554. u2f.sign = function(appId, challenge, registeredKeys, callback, opt_timeoutSeconds) {
  555. if (js_api_version === undefined) {
  556. // Send a message to get the extension to JS API version, then send the actual sign request.
  557. u2f.getApiVersion(
  558. function (response) {
  559. js_api_version = response['js_api_version'] === undefined ? 0 : response['js_api_version'];
  560. console.log("Extension JS API Version: ", js_api_version);
  561. u2f.sendSignRequest(appId, challenge, registeredKeys, callback, opt_timeoutSeconds);
  562. });
  563. } else {
  564. // We know the JS API version. Send the actual sign request in the supported API version.
  565. u2f.sendSignRequest(appId, challenge, registeredKeys, callback, opt_timeoutSeconds);
  566. }
  567. };
  568. /**
  569. * Dispatches an array of sign requests to available U2F tokens.
  570. * @param {string=} appId
  571. * @param {string=} challenge
  572. * @param {Array<u2f.RegisteredKey>} registeredKeys
  573. * @param {function((u2f.Error|u2f.SignResponse))} callback
  574. * @param {number=} opt_timeoutSeconds
  575. */
  576. u2f.sendSignRequest = function(appId, challenge, registeredKeys, callback, opt_timeoutSeconds) {
  577. u2f.getPortSingleton_(function(port) {
  578. var reqId = ++u2f.reqCounter_;
  579. u2f.callbackMap_[reqId] = callback;
  580. var timeoutSeconds = (typeof opt_timeoutSeconds !== 'undefined' ?
  581. opt_timeoutSeconds : u2f.EXTENSION_TIMEOUT_SEC);
  582. var req = u2f.formatSignRequest_(appId, challenge, registeredKeys, timeoutSeconds, reqId);
  583. port.postMessage(req);
  584. });
  585. };
  586. /**
  587. * Dispatches register requests to available U2F tokens. An array of sign
  588. * requests identifies already registered tokens.
  589. * If the JS API version supported by the extension is unknown, it first sends a
  590. * message to the extension to find out the supported API version and then it sends
  591. * the register request.
  592. * @param {string=} appId
  593. * @param {Array<u2f.RegisterRequest>} registerRequests
  594. * @param {Array<u2f.RegisteredKey>} registeredKeys
  595. * @param {function((u2f.Error|u2f.RegisterResponse))} callback
  596. * @param {number=} opt_timeoutSeconds
  597. */
  598. u2f.register = function(appId, registerRequests, registeredKeys, callback, opt_timeoutSeconds) {
  599. if (js_api_version === undefined) {
  600. // Send a message to get the extension to JS API version, then send the actual register request.
  601. u2f.getApiVersion(
  602. function (response) {
  603. js_api_version = response['js_api_version'] === undefined ? 0: response['js_api_version'];
  604. console.log("Extension JS API Version: ", js_api_version);
  605. u2f.sendRegisterRequest(appId, registerRequests, registeredKeys,
  606. callback, opt_timeoutSeconds);
  607. });
  608. } else {
  609. // We know the JS API version. Send the actual register request in the supported API version.
  610. u2f.sendRegisterRequest(appId, registerRequests, registeredKeys,
  611. callback, opt_timeoutSeconds);
  612. }
  613. };
  614. /**
  615. * Dispatches register requests to available U2F tokens. An array of sign
  616. * requests identifies already registered tokens.
  617. * @param {string=} appId
  618. * @param {Array<u2f.RegisterRequest>} registerRequests
  619. * @param {Array<u2f.RegisteredKey>} registeredKeys
  620. * @param {function((u2f.Error|u2f.RegisterResponse))} callback
  621. * @param {number=} opt_timeoutSeconds
  622. */
  623. u2f.sendRegisterRequest = function(appId, registerRequests, registeredKeys, callback, opt_timeoutSeconds) {
  624. u2f.getPortSingleton_(function(port) {
  625. var reqId = ++u2f.reqCounter_;
  626. u2f.callbackMap_[reqId] = callback;
  627. var timeoutSeconds = (typeof opt_timeoutSeconds !== 'undefined' ?
  628. opt_timeoutSeconds : u2f.EXTENSION_TIMEOUT_SEC);
  629. var req = u2f.formatRegisterRequest_(
  630. appId, registeredKeys, registerRequests, timeoutSeconds, reqId);
  631. port.postMessage(req);
  632. });
  633. };
  634. /**
  635. * Dispatches a message to the extension to find out the supported
  636. * JS API version.
  637. * If the user is on a mobile phone and is thus using Google Authenticator instead
  638. * of the Chrome extension, don't send the request and simply return 0.
  639. * @param {function((u2f.Error|u2f.GetJsApiVersionResponse))} callback
  640. * @param {number=} opt_timeoutSeconds
  641. */
  642. u2f.getApiVersion = function(callback, opt_timeoutSeconds) {
  643. u2f.getPortSingleton_(function(port) {
  644. // If we are using Android Google Authenticator or iOS client app,
  645. // do not fire an intent to ask which JS API version to use.
  646. if (port.getPortType) {
  647. var apiVersion;
  648. switch (port.getPortType()) {
  649. case 'WrappedIosPort_':
  650. case 'WrappedAuthenticatorPort_':
  651. apiVersion = 1.1;
  652. break;
  653. default:
  654. apiVersion = 0;
  655. break;
  656. }
  657. callback({ 'js_api_version': apiVersion });
  658. return;
  659. }
  660. var reqId = ++u2f.reqCounter_;
  661. u2f.callbackMap_[reqId] = callback;
  662. var req = {
  663. type: u2f.MessageTypes.U2F_GET_API_VERSION_REQUEST,
  664. timeoutSeconds: (typeof opt_timeoutSeconds !== 'undefined' ?
  665. opt_timeoutSeconds : u2f.EXTENSION_TIMEOUT_SEC),
  666. requestId: reqId
  667. };
  668. port.postMessage(req);
  669. });
  670. };