UNPKG

15.6 kBJavaScriptView Raw
1"use strict";
2var __extends = (this && this.__extends) || (function () {
3 var extendStatics = function (d, b) {
4 extendStatics = Object.setPrototypeOf ||
5 ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
6 function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };
7 return extendStatics(d, b);
8 };
9 return function (d, b) {
10 if (typeof b !== "function" && b !== null)
11 throw new TypeError("Class extends value " + String(b) + " is not a constructor or null");
12 extendStatics(d, b);
13 function __() { this.constructor = d; }
14 d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
15 };
16})();
17var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
18 function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
19 return new (P || (P = Promise))(function (resolve, reject) {
20 function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
21 function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
22 function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
23 step((generator = generator.apply(thisArg, _arguments || [])).next());
24 });
25};
26var __generator = (this && this.__generator) || function (thisArg, body) {
27 var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
28 return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
29 function verb(n) { return function (v) { return step([n, v]); }; }
30 function step(op) {
31 if (f) throw new TypeError("Generator is already executing.");
32 while (_) try {
33 if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
34 if (y = 0, t) op = [op[0] & 2, t.value];
35 switch (op[0]) {
36 case 0: case 1: t = op; break;
37 case 4: _.label++; return { value: op[1], done: false };
38 case 5: _.label++; y = op[1]; op = [0]; continue;
39 case 7: op = _.ops.pop(); _.trys.pop(); continue;
40 default:
41 if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
42 if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
43 if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
44 if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
45 if (t[2]) _.ops.pop();
46 _.trys.pop(); continue;
47 }
48 op = body.call(thisArg, _);
49 } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
50 if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
51 }
52};
53var __importDefault = (this && this.__importDefault) || function (mod) {
54 return (mod && mod.__esModule) ? mod : { "default": mod };
55};
56exports.__esModule = true;
57var hw_transport_1 = __importDefault(require("@ledgerhq/hw-transport"));
58var hid_framing_1 = __importDefault(require("@ledgerhq/devices/lib/hid-framing"));
59var devices_1 = require("@ledgerhq/devices");
60var logs_1 = require("@ledgerhq/logs");
61var errors_1 = require("@ledgerhq/errors");
62var webusb_1 = require("./webusb");
63var configurationValue = 1;
64var endpointNumber = 3;
65/**
66 * WebUSB Transport implementation
67 * @example
68 * import TransportWebUSB from "@ledgerhq/hw-transport-webusb";
69 * ...
70 * TransportWebUSB.create().then(transport => ...)
71 */
72var TransportWebUSB = /** @class */ (function (_super) {
73 __extends(TransportWebUSB, _super);
74 function TransportWebUSB(device, interfaceNumber) {
75 var _this = _super.call(this) || this;
76 _this.channel = Math.floor(Math.random() * 0xffff);
77 _this.packetSize = 64;
78 _this._disconnectEmitted = false;
79 _this._emitDisconnect = function (e) {
80 if (_this._disconnectEmitted)
81 return;
82 _this._disconnectEmitted = true;
83 _this.emit("disconnect", e);
84 };
85 _this.device = device;
86 _this.interfaceNumber = interfaceNumber;
87 _this.deviceModel = (0, devices_1.identifyUSBProductId)(device.productId);
88 return _this;
89 }
90 /**
91 * Similar to create() except it will always display the device permission (even if some devices are already accepted).
92 */
93 TransportWebUSB.request = function () {
94 return __awaiter(this, void 0, void 0, function () {
95 var device;
96 return __generator(this, function (_a) {
97 switch (_a.label) {
98 case 0: return [4 /*yield*/, (0, webusb_1.requestLedgerDevice)()];
99 case 1:
100 device = _a.sent();
101 return [2 /*return*/, TransportWebUSB.open(device)];
102 }
103 });
104 });
105 };
106 /**
107 * Similar to create() except it will never display the device permission (it returns a Promise<?Transport>, null if it fails to find a device).
108 */
109 TransportWebUSB.openConnected = function () {
110 return __awaiter(this, void 0, void 0, function () {
111 var devices;
112 return __generator(this, function (_a) {
113 switch (_a.label) {
114 case 0: return [4 /*yield*/, (0, webusb_1.getLedgerDevices)()];
115 case 1:
116 devices = _a.sent();
117 if (devices.length === 0)
118 return [2 /*return*/, null];
119 return [2 /*return*/, TransportWebUSB.open(devices[0])];
120 }
121 });
122 });
123 };
124 /**
125 * Create a Ledger transport with a USBDevice
126 */
127 TransportWebUSB.open = function (device) {
128 return __awaiter(this, void 0, void 0, function () {
129 var iface, interfaceNumber, e_1, transport, onDisconnect;
130 return __generator(this, function (_a) {
131 switch (_a.label) {
132 case 0: return [4 /*yield*/, device.open()];
133 case 1:
134 _a.sent();
135 if (!(device.configuration === null)) return [3 /*break*/, 3];
136 return [4 /*yield*/, device.selectConfiguration(configurationValue)];
137 case 2:
138 _a.sent();
139 _a.label = 3;
140 case 3: return [4 /*yield*/, gracefullyResetDevice(device)];
141 case 4:
142 _a.sent();
143 iface = device.configurations[0].interfaces.find(function (_a) {
144 var alternates = _a.alternates;
145 return alternates.some(function (a) { return a.interfaceClass === 255; });
146 });
147 if (!iface) {
148 throw new errors_1.TransportInterfaceNotAvailable("No WebUSB interface found for your Ledger device. Please upgrade firmware or contact techsupport.");
149 }
150 interfaceNumber = iface.interfaceNumber;
151 _a.label = 5;
152 case 5:
153 _a.trys.push([5, 7, , 9]);
154 return [4 /*yield*/, device.claimInterface(interfaceNumber)];
155 case 6:
156 _a.sent();
157 return [3 /*break*/, 9];
158 case 7:
159 e_1 = _a.sent();
160 return [4 /*yield*/, device.close()];
161 case 8:
162 _a.sent();
163 throw new errors_1.TransportInterfaceNotAvailable(e_1.message);
164 case 9:
165 transport = new TransportWebUSB(device, interfaceNumber);
166 onDisconnect = function (e) {
167 if (device === e.device) {
168 // $FlowFixMe
169 navigator.usb.removeEventListener("disconnect", onDisconnect);
170 transport._emitDisconnect(new errors_1.DisconnectedDevice());
171 }
172 };
173 // $FlowFixMe
174 navigator.usb.addEventListener("disconnect", onDisconnect);
175 return [2 /*return*/, transport];
176 }
177 });
178 });
179 };
180 /**
181 * Release the transport device
182 */
183 TransportWebUSB.prototype.close = function () {
184 return __awaiter(this, void 0, void 0, function () {
185 return __generator(this, function (_a) {
186 switch (_a.label) {
187 case 0: return [4 /*yield*/, this.exchangeBusyPromise];
188 case 1:
189 _a.sent();
190 return [4 /*yield*/, this.device.releaseInterface(this.interfaceNumber)];
191 case 2:
192 _a.sent();
193 return [4 /*yield*/, gracefullyResetDevice(this.device)];
194 case 3:
195 _a.sent();
196 return [4 /*yield*/, this.device.close()];
197 case 4:
198 _a.sent();
199 return [2 /*return*/];
200 }
201 });
202 });
203 };
204 /**
205 * Exchange with the device using APDU protocol.
206 * @param apdu
207 * @returns a promise of apdu response
208 */
209 TransportWebUSB.prototype.exchange = function (apdu) {
210 return __awaiter(this, void 0, void 0, function () {
211 var b;
212 var _this = this;
213 return __generator(this, function (_a) {
214 switch (_a.label) {
215 case 0: return [4 /*yield*/, this.exchangeAtomicImpl(function () { return __awaiter(_this, void 0, void 0, function () {
216 var _a, channel, packetSize, framing, blocks, i, result, acc, r, buffer;
217 return __generator(this, function (_b) {
218 switch (_b.label) {
219 case 0:
220 _a = this, channel = _a.channel, packetSize = _a.packetSize;
221 (0, logs_1.log)("apdu", "=> " + apdu.toString("hex"));
222 framing = (0, hid_framing_1["default"])(channel, packetSize);
223 blocks = framing.makeBlocks(apdu);
224 i = 0;
225 _b.label = 1;
226 case 1:
227 if (!(i < blocks.length)) return [3 /*break*/, 4];
228 return [4 /*yield*/, this.device.transferOut(endpointNumber, blocks[i])];
229 case 2:
230 _b.sent();
231 _b.label = 3;
232 case 3:
233 i++;
234 return [3 /*break*/, 1];
235 case 4:
236 if (!!(result = framing.getReducedResult(acc))) return [3 /*break*/, 6];
237 return [4 /*yield*/, this.device.transferIn(endpointNumber, packetSize)];
238 case 5:
239 r = _b.sent();
240 buffer = Buffer.from(r.data.buffer);
241 acc = framing.reduceResponse(acc, buffer);
242 return [3 /*break*/, 4];
243 case 6:
244 (0, logs_1.log)("apdu", "<= " + result.toString("hex"));
245 return [2 /*return*/, result];
246 }
247 });
248 }); })["catch"](function (e) {
249 if (e && e.message && e.message.includes("disconnected")) {
250 _this._emitDisconnect(e);
251 throw new errors_1.DisconnectedDeviceDuringOperation(e.message);
252 }
253 throw e;
254 })];
255 case 1:
256 b = _a.sent();
257 return [2 /*return*/, b];
258 }
259 });
260 });
261 };
262 TransportWebUSB.prototype.setScrambleKey = function () { };
263 /**
264 * Check if WebUSB transport is supported.
265 */
266 TransportWebUSB.isSupported = webusb_1.isSupported;
267 /**
268 * List the WebUSB devices that was previously authorized by the user.
269 */
270 TransportWebUSB.list = webusb_1.getLedgerDevices;
271 /**
272 * Actively listen to WebUSB devices and emit ONE device
273 * that was either accepted before, if not it will trigger the native permission UI.
274 *
275 * Important: it must be called in the context of a UI click!
276 */
277 TransportWebUSB.listen = function (observer) {
278 var unsubscribed = false;
279 (0, webusb_1.getFirstLedgerDevice)().then(function (device) {
280 if (!unsubscribed) {
281 var deviceModel = (0, devices_1.identifyUSBProductId)(device.productId);
282 observer.next({
283 type: "add",
284 descriptor: device,
285 deviceModel: deviceModel
286 });
287 observer.complete();
288 }
289 }, function (error) {
290 if (window.DOMException &&
291 error instanceof window.DOMException &&
292 error.code === 18) {
293 observer.error(new errors_1.TransportWebUSBGestureRequired(error.message));
294 }
295 else {
296 observer.error(new errors_1.TransportOpenUserCancelled(error.message));
297 }
298 });
299 function unsubscribe() {
300 unsubscribed = true;
301 }
302 return {
303 unsubscribe: unsubscribe
304 };
305 };
306 return TransportWebUSB;
307}(hw_transport_1["default"]));
308exports["default"] = TransportWebUSB;
309function gracefullyResetDevice(device) {
310 return __awaiter(this, void 0, void 0, function () {
311 var err_1;
312 return __generator(this, function (_a) {
313 switch (_a.label) {
314 case 0:
315 _a.trys.push([0, 2, , 3]);
316 return [4 /*yield*/, device.reset()];
317 case 1:
318 _a.sent();
319 return [3 /*break*/, 3];
320 case 2:
321 err_1 = _a.sent();
322 console.warn(err_1);
323 return [3 /*break*/, 3];
324 case 3: return [2 /*return*/];
325 }
326 });
327 });
328}
329//# sourceMappingURL=TransportWebUSB.js.map
\No newline at end of file