@ledgerhq/device-transport-kit-react-native-hid 0.0.0-hid-candidate-2-20250526084932 → 0.0.0-hid-candidate-4-20250528125313

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -49,7 +49,7 @@ class TransportHidModule(
49
49
  private val loggerService: LoggerService =
50
50
  LoggerService { info ->
51
51
  Timber.tag("LDMKTransportHIDModule " + info.tag).d(info.message)
52
- sendEvent(reactContext, BridgeEvents.TransportLog(info))
52
+ // sendEvent(reactContext, BridgeEvents.TransportLog(info))
53
53
  }
54
54
 
55
55
  private val transport: AndroidUsbTransport? by lazy {
@@ -187,19 +187,30 @@ internal class DefaultAndroidUsbTransport(
187
187
  "Reconnecting device (sessionId=${deviceConnection.sessionId})"
188
188
  )
189
189
  )
190
- deviceConnection.handleDeviceConnected(
191
- AndroidUsbApduSender(
192
- dependencies = AndroidUsbApduSender.Dependencies(
193
- usbDevice = usbDevice,
194
- ledgerUsbDevice = state.ledgerUsbDevice,
195
- ),
196
- usbManager = usbManager,
197
- ioDispatcher = Dispatchers.IO,
198
- framerService = FramerService(loggerService),
199
- request = UsbRequest(),
200
- loggerService = loggerService
201
- )
190
+
191
+ val apduSender = AndroidUsbApduSender(
192
+ dependencies = AndroidUsbApduSender.Dependencies(
193
+ usbDevice = usbDevice,
194
+ ledgerUsbDevice = state.ledgerUsbDevice,
195
+ ),
196
+ usbManager = usbManager,
197
+ ioDispatcher = Dispatchers.IO,
198
+ framerService = FramerService(loggerService),
199
+ request = UsbRequest(),
200
+ loggerService = loggerService
202
201
  )
202
+
203
+ if (!usbConnectionsPendingReconnection.contains(deviceConnection)) {
204
+ /**
205
+ * We check this because maybe by the time we get here,
206
+ * the reconnection has timed out and the session has been terminated.
207
+ * Easy to reproduce for instance if the permission request requires
208
+ * a user interaction.
209
+ */
210
+ apduSender.release()
211
+ return@launch
212
+ }
213
+ deviceConnection.handleDeviceConnected(apduSender)
203
214
  usbConnectionsPendingReconnection.remove(deviceConnection)
204
215
  usbConnections[deviceConnection.sessionId] = deviceConnection
205
216
  }
@@ -335,7 +346,9 @@ internal class DefaultAndroidUsbTransport(
335
346
  }
336
347
 
337
348
  override suspend fun disconnect(deviceId: String) {
349
+ // The DeviceConnection is either in usbConnections or usbConnectionsPendingReconnection
338
350
  usbConnections[deviceId]?.requestCloseConnection()
351
+ usbConnectionsPendingReconnection.find { it.sessionId == deviceId }?.requestCloseConnection()
339
352
  }
340
353
 
341
354
  private fun generateSessionId(usbDevice: UsbDevice): String = "usb_${usbDevice.deviceId}"
@@ -3,6 +3,7 @@ package com.ledger.devicesdk.shared.androidMainInternal.transport.deviceconnecti
3
3
  import com.ledger.devicesdk.shared.api.apdu.SendApduFailureReason
4
4
  import com.ledger.devicesdk.shared.api.apdu.SendApduResult
5
5
  import com.ledger.devicesdk.shared.internal.service.logger.LoggerService
6
+ import com.ledger.devicesdk.shared.internal.service.logger.buildSimpleDebugLogInfo
6
7
  import kotlinx.coroutines.CoroutineDispatcher
7
8
  import kotlinx.coroutines.CoroutineScope
8
9
  import kotlinx.coroutines.Job
@@ -40,11 +41,16 @@ internal class DeviceConnectionStateMachine(
40
41
  }
41
42
  }
42
43
  this.state = newState
44
+ loggerService.log(buildSimpleDebugLogInfo("DeviceConnectionStateMachine", "-> New state: $newState"))
43
45
  }
44
46
 
45
47
  private fun handleEvent(event: Event) {
46
- val currentState = state
47
- when (currentState) {
48
+ val logMessage = """
49
+ -> Event received: $event
50
+ In state: $state
51
+ """.trimIndent()
52
+ loggerService.log(buildSimpleDebugLogInfo("DeviceConnectionStateMachine", logMessage))
53
+ when (val currentState = state) {
48
54
  is State.Connected -> {
49
55
  when (event) {
50
56
  is Event.SendApduRequested -> {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/api/transport/RNHidTransport.test.ts"],
4
- "sourcesContent": ["import {\n DeviceModelId,\n DisconnectError,\n type LoggerPublisherService,\n LogLevel,\n type LogParams,\n OpeningConnectionError,\n StaticDeviceModelDataSource,\n TransportConnectedDevice,\n type TransportDiscoveredDevice,\n} from \"@ledgerhq/device-management-kit\";\nimport { Left, Right } from \"purify-ts/Either\";\nimport { Subject } from \"rxjs\";\n\nimport { TRANSPORT_IDENTIFIER } from \"@api/transport/rnHidTransportIdentifier\";\n\nimport { HidTransportSendApduUnknownError } from \"./Errors\";\nimport { type NativeModuleWrapper } from \"./NativeModuleWrapper\";\nimport { RNHidTransport } from \"./RNHidTransport\";\nimport { type InternalDeviceDisconnected } from \"./types\";\n\nconst deviceModelDataSource = new StaticDeviceModelDataSource();\n\nconst makeMockTransportDiscoveredDevice = (\n id: string,\n deviceModelId: DeviceModelId,\n): TransportDiscoveredDevice => ({\n id,\n transport: TRANSPORT_IDENTIFIER,\n name: deviceModelId,\n deviceModel: deviceModelDataSource.getDeviceModel({\n id: deviceModelId,\n }),\n});\n\nconst mockDiscoveredDevice1 = makeMockTransportDiscoveredDevice(\n \"1\",\n DeviceModelId.NANO_S,\n);\nconst mockDiscoveredDevice2 = makeMockTransportDiscoveredDevice(\n \"2\",\n DeviceModelId.NANO_X,\n);\nconst mockDiscoveredDevice3 = makeMockTransportDiscoveredDevice(\n \"3\",\n DeviceModelId.NANO_S,\n);\n\ndescribe(\"RNHidTransport\", () => {\n let discoveredDevicesSubject: Subject<Array<TransportDiscoveredDevice>>;\n let deviceDisconnectedSubject: Subject<InternalDeviceDisconnected>;\n let transportLogsSubject: Subject<LogParams>;\n let nativeModuleWrapper: NativeModuleWrapper;\n const fakeLogger: LoggerPublisherService = {\n error: vi.fn(),\n warn: vi.fn(),\n info: vi.fn(),\n debug: vi.fn(),\n subscribers: [],\n };\n const loggerServiceFactory = vi.fn(() => fakeLogger);\n\n const wrapperStartScan = vi.fn();\n const wrapperStopScan = vi.fn();\n const wrapperConnectDevice = vi.fn();\n const wrapperDisconnectDevice = vi.fn();\n const wrapperSendApdu = vi.fn();\n\n beforeEach(() => {\n discoveredDevicesSubject = new Subject();\n deviceDisconnectedSubject = new Subject();\n transportLogsSubject = new Subject();\n\n vi.clearAllMocks();\n\n wrapperStartScan.mockResolvedValue(undefined);\n wrapperStopScan.mockResolvedValue(undefined);\n wrapperConnectDevice.mockResolvedValue(undefined);\n wrapperDisconnectDevice.mockResolvedValue(undefined);\n wrapperSendApdu.mockResolvedValue(undefined);\n\n nativeModuleWrapper = {\n startScan: wrapperStartScan,\n stopScan: wrapperStopScan,\n subscribeToDiscoveredDevicesEvents: vi.fn(() =>\n discoveredDevicesSubject.asObservable(),\n ),\n subscribeToDeviceDisconnectedEvents: vi.fn(() =>\n deviceDisconnectedSubject.asObservable(),\n ),\n subscribeToTransportLogs: vi.fn(() =>\n transportLogsSubject.asObservable(),\n ),\n connectDevice: wrapperConnectDevice,\n disconnectDevice: wrapperDisconnectDevice,\n sendApdu: wrapperSendApdu,\n };\n });\n\n test(\"getIdentifier returns TRANSPORT_IDENTIFIER\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transport.getIdentifier()).toBe(TRANSPORT_IDENTIFIER);\n });\n\n describe(\"isSupported returns the provided support flag\", () => {\n test(\"supported\", () => {\n // given\n const transportTrue = new RNHidTransport(\n true, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportTrue.isSupported()).toBe(true);\n });\n\n test(\"not supported\", () => {\n // given\n const transportFalse = new RNHidTransport(\n false, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportFalse.isSupported()).toBe(false);\n });\n });\n\n test(\"constructor subscribes to transport logs and calls logger methods\", () => {\n // given\n new RNHidTransport(true, nativeModuleWrapper, loggerServiceFactory);\n\n // when\n const logEvent: LogParams = [\n LogLevel.Info,\n \"Test message\",\n {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n },\n ];\n transportLogsSubject.next(logEvent);\n\n // then\n expect(fakeLogger.info).toHaveBeenCalledWith(\"Test message\", {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n });\n });\n\n describe(\"startDiscovering\", () => {\n it(\"calls startScan\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n transport.startDiscovering().subscribe();\n\n // then\n expect(wrapperStartScan).toHaveBeenCalled();\n });\n\n it(\"emits new discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const discoveredDevices: TransportDiscoveredDevice[] = [];\n\n // when\n transport.startDiscovering().subscribe({\n next: (device) => {\n discoveredDevices.push(device);\n if (discoveredDevices.length === 3) {\n try {\n // then\n expect(discoveredDevices).toEqual([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n error: reject,\n });\n // Simulate getObservableOfArraysNewItems behavior:\n // emit first array with one device then the subsequent arrays with a new device.\n discoveredDevicesSubject.next([mockDiscoveredDevice1]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.startDiscovering().subscribe({\n next: () => {},\n error: (err) => {\n try {\n // then\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n });\n\n describe(\"stopDiscovering\", () => {\n it(\"calls stopScan\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n await transport.stopDiscovering();\n\n // then\n expect(nativeModuleWrapper.stopScan).toHaveBeenCalled();\n });\n\n it(\"logs error when stopScan fails\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n\n // then\n await transport.stopDiscovering();\n expect(fakeLogger.error).toHaveBeenCalledWith(\n \"stopDiscovering error\",\n error,\n );\n });\n });\n\n describe(\"listenToKnownDevices\", () => {\n it(\"emits arrays of discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const observedEvents: Array<Array<TransportDiscoveredDevice>> = [];\n\n // when\n transport.listenToAvailableDevices().subscribe({\n next: (devices) => {\n observedEvents.push(devices);\n if (observedEvents.length === 2) {\n try {\n // then\n expect(observedEvents).toEqual([\n [mockDiscoveredDevice1, mockDiscoveredDevice2],\n [mockDiscoveredDevice1, mockDiscoveredDevice3],\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n complete: () => {\n reject(\"should not complete\");\n },\n error: reject,\n });\n\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"start scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.listenToAvailableDevices().subscribe({\n error: (err) => {\n // then\n try {\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n\n it(\"calls stopScan on unsubscribe\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop scan failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n const subscription = transport.listenToAvailableDevices().subscribe({});\n subscription.unsubscribe();\n\n expect(wrapperStopScan).toHaveBeenCalled();\n });\n });\n\n describe(\"connect\", () => {\n describe(\"connection successful\", () => {\n it(\"should return a Right(TransportConnectedDevice) on successful connection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result.isRight()).toBe(true);\n const connectedDevice = result.extract() as TransportConnectedDevice;\n expect(connectedDevice).toBeInstanceOf(TransportConnectedDevice);\n expect(connectedDevice.id).toBe(sessionId);\n expect(connectedDevice.deviceModel).toEqual(fakeDeviceModel);\n expect(connectedDevice.transport).toBe(TRANSPORT_IDENTIFIER);\n expect(connectedDevice.type).toBe(\"USB\");\n });\n\n test(\"should trigger onDisconnect when a matching disconnect event is emitted\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const onDisconnect = vi.fn();\n\n // when\n await transport.connect({\n deviceId: sessionId,\n onDisconnect,\n });\n deviceDisconnectedSubject.next({ sessionId });\n\n // then\n expect(onDisconnect).toHaveBeenCalledWith(sessionId);\n });\n\n test(\"should handle sendApdu success (Right)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Right(\"apduResponse\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu, false, 0);\n\n // then\n expect(nativeModuleWrapper.sendApdu).toHaveBeenCalledWith(\n sessionId,\n apdu,\n false,\n 0,\n );\n expect(apduResult).toEqual(Right(\"apduResponse\"));\n });\n\n test(\"should handle sendApdu failure (Left)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Left(\"some error\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu);\n\n // then\n expect(apduResult).toEqual(Left(\"some error\"));\n });\n\n test(\"should handle sendApdu rejection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n const apduError = new Error(\"apdu failed\");\n\n // when\n wrapperSendApdu.mockRejectedValueOnce(apduError);\n const apduResult = await connectedDevice.sendApdu(new Uint8Array([]));\n\n // then\n expect(apduResult).toEqual(Left(new HidTransportSendApduUnknownError(apduError)));\n });\n });\n\n describe(\"connection failure\", () => {\n test(\"should return a Left when nativeModuleWrapper.connectDevice resolves a Left\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const errorResult = Left(\n new OpeningConnectionError(\"connection failed\"),\n );\n wrapperConnectDevice.mockResolvedValueOnce(errorResult);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(errorResult);\n });\n\n test(\"should return a Left when nativeModuleWrapper.connectDevice rejects\", async () => {\n // given\n const error = new Error(\"connection failed\");\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n wrapperConnectDevice.mockRejectedValueOnce(error);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(Left(new OpeningConnectionError(error)));\n });\n });\n });\n\n describe(\"disconnect\", () => {\n it(\"returns Right on successful disconnect\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.disconnect({\n connectedDevice: { id: \"session789\" } as TransportConnectedDevice,\n });\n\n // then\n expect(nativeModuleWrapper.disconnectDevice).toHaveBeenCalledWith(\n \"session789\",\n );\n expect(result).toEqual(Right(undefined));\n });\n\n it(\"returns Left on disconnect failure\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"disconnect failed\");\n wrapperDisconnectDevice.mockRejectedValueOnce(error);\n const result = await transport.disconnect({\n connectedDevice: { id: \"session000\" } as TransportConnectedDevice,\n });\n\n // then\n expect(result).toEqual(Left(new DisconnectError(error)));\n });\n });\n});\n"],
5
- "mappings": "aAAA,IAAAA,EAUO,2CACPC,EAA4B,4BAC5BC,EAAwB,gBAExBC,EAAqC,mDAErCC,EAAiD,oBAEjDC,EAA+B,4BAG/B,MAAMC,EAAwB,IAAI,8BAE5BC,EAAoC,CACxCC,EACAC,KAC+B,CAC/B,GAAAD,EACA,UAAW,uBACX,KAAMC,EACN,YAAaH,EAAsB,eAAe,CAChD,GAAIG,CACN,CAAC,CACH,GAEMC,EAAwBH,EAC5B,IACA,gBAAc,MAChB,EACMI,EAAwBJ,EAC5B,IACA,gBAAc,MAChB,EACMK,EAAwBL,EAC5B,IACA,gBAAc,MAChB,EAEA,SAAS,iBAAkB,IAAM,CAC/B,IAAIM,EACAC,EACAC,EACAC,EACJ,MAAMC,EAAqC,CACzC,MAAO,GAAG,GAAG,EACb,KAAM,GAAG,GAAG,EACZ,KAAM,GAAG,GAAG,EACZ,MAAO,GAAG,GAAG,EACb,YAAa,CAAC,CAChB,EACMC,EAAuB,GAAG,GAAG,IAAMD,CAAU,EAE7CE,EAAmB,GAAG,GAAG,EACzBC,EAAkB,GAAG,GAAG,EACxBC,EAAuB,GAAG,GAAG,EAC7BC,EAA0B,GAAG,GAAG,EAChCC,EAAkB,GAAG,GAAG,EAE9B,WAAW,IAAM,CACfV,EAA2B,IAAI,UAC/BC,EAA4B,IAAI,UAChCC,EAAuB,IAAI,UAE3B,GAAG,cAAc,EAEjBI,EAAiB,kBAAkB,MAAS,EAC5CC,EAAgB,kBAAkB,MAAS,EAC3CC,EAAqB,kBAAkB,MAAS,EAChDC,EAAwB,kBAAkB,MAAS,EACnDC,EAAgB,kBAAkB,MAAS,EAE3CP,EAAsB,CACpB,UAAWG,EACX,SAAUC,EACV,mCAAoC,GAAG,GAAG,IACxCP,EAAyB,aAAa,CACxC,EACA,oCAAqC,GAAG,GAAG,IACzCC,EAA0B,aAAa,CACzC,EACA,yBAA0B,GAAG,GAAG,IAC9BC,EAAqB,aAAa,CACpC,EACA,cAAeM,EACf,iBAAkBC,EAClB,SAAUC,CACZ,CACF,CAAC,EAED,KAAK,6CAA8C,IAAM,CAEvD,MAAMC,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAEA,OAAOM,EAAU,cAAc,CAAC,EAAE,KAAK,sBAAoB,CAC7D,CAAC,EAED,SAAS,gDAAiD,IAAM,CAC9D,KAAK,YAAa,IAAM,CAEtB,MAAMC,EAAgB,IAAI,iBACxB,GACAT,EACAE,CACF,EAEA,OAAOO,EAAc,YAAY,CAAC,EAAE,KAAK,EAAI,CAC/C,CAAC,EAED,KAAK,gBAAiB,IAAM,CAE1B,MAAMC,EAAiB,IAAI,iBACzB,GACAV,EACAE,CACF,EAEA,OAAOQ,EAAe,YAAY,CAAC,EAAE,KAAK,EAAK,CACjD,CAAC,CACH,CAAC,EAED,KAAK,oEAAqE,IAAM,CAE9E,IAAI,iBAAe,GAAMV,EAAqBE,CAAoB,EAGlE,MAAMS,EAAsB,CAC1B,WAAS,KACT,eACA,CACE,IAAK,UACL,KAAM,CAAE,IAAK,OAAQ,EACrB,UAAW,SACb,CACF,EACAZ,EAAqB,KAAKY,CAAQ,EAGlC,OAAOV,EAAW,IAAI,EAAE,qBAAqB,eAAgB,CAC3D,IAAK,UACL,KAAM,CAAE,IAAK,OAAQ,EACrB,UAAW,SACb,CAAC,CACH,CAAC,EAED,SAAS,mBAAoB,IAAM,CACjC,GAAG,kBAAmB,IAAM,CAER,IAAI,iBACpB,GACAD,EACAE,CACF,EAGU,iBAAiB,EAAE,UAAU,EAGvC,OAAOC,CAAgB,EAAE,iBAAiB,CAC5C,CAAC,EAED,GAAG,+BAAgC,IAC1B,IAAI,QAAc,CAACS,EAASC,IAAW,CAE5C,MAAML,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EACMY,EAAiD,CAAC,EAGxDN,EAAU,iBAAiB,EAAE,UAAU,CACrC,KAAOO,GAAW,CAEhB,GADAD,EAAkB,KAAKC,CAAM,EACzBD,EAAkB,SAAW,EAC/B,GAAI,CAEF,OAAOA,CAAiB,EAAE,QAAQ,CAChCpB,EACAC,EACAC,CACF,CAAC,EACDgB,EAAQ,CACV,OAASI,EAAG,CACVH,EAAOG,CAAC,CACV,CAEJ,EACA,MAAOH,CACT,CAAC,EAGDhB,EAAyB,KAAK,CAACH,CAAqB,CAAC,EACrDG,EAAyB,KAAK,CAC5BH,EACAC,CACF,CAAC,EACDE,EAAyB,KAAK,CAC5BH,EACAC,EACAC,CACF,CAAC,CACH,CAAC,CACF,EAED,GAAG,6BAA8B,IACxB,IAAI,QAAc,CAACgB,EAASC,IAAW,CAE5C,MAAML,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGMe,EAAQ,IAAI,MAAM,aAAa,EACrCd,EAAiB,sBAAsBc,CAAK,EAC5CT,EAAU,iBAAiB,EAAE,UAAU,CACrC,KAAM,IAAM,CAAC,EACb,MAAQU,GAAQ,CACd,GAAI,CAEF,OAAOA,CAAG,EAAE,KAAKD,CAAK,EACtBL,EAAQ,CACV,OAASI,EAAG,CACVH,EAAOG,CAAC,CACV,CACF,CACF,CAAC,CACH,CAAC,CACF,CACH,CAAC,EAED,SAAS,kBAAmB,IAAM,CAChC,GAAG,iBAAkB,SAAY,CAS/B,MAPkB,IAAI,iBACpB,GACAhB,EACAE,CACF,EAGgB,gBAAgB,EAGhC,OAAOF,EAAoB,QAAQ,EAAE,iBAAiB,CACxD,CAAC,EAED,GAAG,iCAAkC,SAAY,CAE/C,MAAMQ,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGMe,EAAQ,IAAI,MAAM,aAAa,EACrCb,EAAgB,sBAAsBa,CAAK,EAG3C,MAAMT,EAAU,gBAAgB,EAChC,OAAOP,EAAW,KAAK,EAAE,qBACvB,wBACAgB,CACF,CACF,CAAC,CACH,CAAC,EAED,SAAS,uBAAwB,IAAM,CACrC,GAAG,qCAAsC,IAChC,IAAI,QAAc,CAACL,EAASC,IAAW,CAE5C,MAAML,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EACMiB,EAA0D,CAAC,EAGjEX,EAAU,yBAAyB,EAAE,UAAU,CAC7C,KAAOY,GAAY,CAEjB,GADAD,EAAe,KAAKC,CAAO,EACvBD,EAAe,SAAW,EAC5B,GAAI,CAEF,OAAOA,CAAc,EAAE,QAAQ,CAC7B,CAACzB,EAAuBC,CAAqB,EAC7C,CAACD,EAAuBE,CAAqB,CAC/C,CAAC,EACDgB,EAAQ,CACV,OAASI,EAAG,CACVH,EAAOG,CAAC,CACV,CAEJ,EACA,SAAU,IAAM,CACdH,EAAO,qBAAqB,CAC9B,EACA,MAAOA,CACT,CAAC,EAEDhB,EAAyB,KAAK,CAC5BH,EACAC,CACF,CAAC,EACDE,EAAyB,KAAK,CAC5BH,EACAE,CACF,CAAC,CACH,CAAC,CACF,EAED,GAAG,6BAA8B,IACxB,IAAI,QAAc,CAACgB,EAASC,IAAW,CAE5C,MAAML,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGMe,EAAQ,IAAI,MAAM,mBAAmB,EAC3Cd,EAAiB,sBAAsBc,CAAK,EAC5CT,EAAU,yBAAyB,EAAE,UAAU,CAC7C,MAAQU,GAAQ,CAEd,GAAI,CACF,OAAOA,CAAG,EAAE,KAAKD,CAAK,EACtBL,EAAQ,CACV,OAASI,EAAG,CACVH,EAAOG,CAAC,CACV,CACF,CACF,CAAC,CACH,CAAC,CACF,EAED,GAAG,gCAAiC,IAAM,CAExC,MAAMR,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGMe,EAAQ,IAAI,MAAM,kBAAkB,EAC1Cb,EAAgB,sBAAsBa,CAAK,EACtBT,EAAU,yBAAyB,EAAE,UAAU,CAAC,CAAC,EACzD,YAAY,EAEzB,OAAOJ,CAAe,EAAE,iBAAiB,CAC3C,CAAC,CACH,CAAC,EAED,SAAS,UAAW,IAAM,CACxB,SAAS,wBAAyB,IAAM,CACtC,GAAG,2EAA4E,SAAY,CAEzF,MAAMiB,EAAkB,CAAE,MAAO,WAAY,EACvCC,EAAY,aAClBjB,EAAqB,yBACnB,SAAM,CAAE,UAAAiB,EAAW,qBAAsBD,CAAgB,CAAC,CAC5D,EAQA,MAAME,EAAS,MAPG,IAAI,iBACpB,GACAvB,EACAE,CACF,EAG+B,QAAQ,CACrC,SAAUoB,EACV,aAAc,GAAG,GAAG,CACtB,CAAC,EAGD,OAAOC,EAAO,QAAQ,CAAC,EAAE,KAAK,EAAI,EAClC,MAAMC,EAAkBD,EAAO,QAAQ,EACvC,OAAOC,CAAe,EAAE,eAAe,0BAAwB,EAC/D,OAAOA,EAAgB,EAAE,EAAE,KAAKF,CAAS,EACzC,OAAOE,EAAgB,WAAW,EAAE,QAAQH,CAAe,EAC3D,OAAOG,EAAgB,SAAS,EAAE,KAAK,sBAAoB,EAC3D,OAAOA,EAAgB,IAAI,EAAE,KAAK,KAAK,CACzC,CAAC,EAED,KAAK,0EAA2E,SAAY,CAE1F,MAAMH,EAAkB,CAAE,MAAO,WAAY,EACvCC,EAAY,aAClBjB,EAAqB,yBACnB,SAAM,CAAE,UAAAiB,EAAW,qBAAsBD,CAAgB,CAAC,CAC5D,EACA,MAAMb,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EACMuB,EAAe,GAAG,GAAG,EAG3B,MAAMjB,EAAU,QAAQ,CACtB,SAAUc,EACV,aAAAG,CACF,CAAC,EACD3B,EAA0B,KAAK,CAAE,UAAAwB,CAAU,CAAC,EAG5C,OAAOG,CAAY,EAAE,qBAAqBH,CAAS,CACrD,CAAC,EAED,KAAK,yCAA0C,SAAY,CAEzD,MAAMD,EAAkB,CAAE,MAAO,WAAY,EACvCC,EAAY,aAClBjB,EAAqB,yBACnB,SAAM,CAAE,UAAAiB,EAAW,qBAAsBD,CAAgB,CAAC,CAC5D,EAUA,MAAMG,GAJS,MALG,IAAI,iBACpB,GACAxB,EACAE,CACF,EAC+B,QAAQ,CACrC,SAAUoB,EACV,aAAc,GAAG,GAAG,CACtB,CAAC,GAC8B,QAAQ,EAGvCf,EAAgB,yBAAsB,SAAM,cAAc,CAAC,EAC3D,MAAMmB,EAAO,IAAI,WAAW,CAAC,EAAG,EAAG,CAAC,CAAC,EAC/BC,EAAa,MAAMH,EAAgB,SAASE,EAAM,GAAO,CAAC,EAGhE,OAAO1B,EAAoB,QAAQ,EAAE,qBACnCsB,EACAI,EACA,GACA,CACF,EACA,OAAOC,CAAU,EAAE,WAAQ,SAAM,cAAc,CAAC,CAClD,CAAC,EAED,KAAK,wCAAyC,SAAY,CAExD,MAAMN,EAAkB,CAAE,MAAO,WAAY,EACvCC,EAAY,aAClBjB,EAAqB,yBACnB,SAAM,CAAE,UAAAiB,EAAW,qBAAsBD,CAAgB,CAAC,CAC5D,EAUA,MAAMG,GAJS,MALG,IAAI,iBACpB,GACAxB,EACAE,CACF,EAC+B,QAAQ,CACrC,SAAUoB,EACV,aAAc,GAAG,GAAG,CACtB,CAAC,GAC8B,QAAQ,EAGvCf,EAAgB,yBAAsB,QAAK,YAAY,CAAC,EACxD,MAAMmB,EAAO,IAAI,WAAW,CAAC,EAAG,EAAG,CAAC,CAAC,EAC/BC,EAAa,MAAMH,EAAgB,SAASE,CAAI,EAGtD,OAAOC,CAAU,EAAE,WAAQ,QAAK,YAAY,CAAC,CAC/C,CAAC,EAED,KAAK,mCAAoC,SAAY,CAEnD,MAAMN,EAAkB,CAAE,MAAO,WAAY,EACvCC,EAAY,aAClBjB,EAAqB,yBACnB,SAAM,CAAE,UAAAiB,EAAW,qBAAsBD,CAAgB,CAAC,CAC5D,EAUA,MAAMG,GAJS,MALG,IAAI,iBACpB,GACAxB,EACAE,CACF,EAC+B,QAAQ,CACrC,SAAUoB,EACV,aAAc,GAAG,GAAG,CACtB,CAAC,GAC8B,QAAQ,EACjCM,EAAY,IAAI,MAAM,aAAa,EAGzCrB,EAAgB,sBAAsBqB,CAAS,EAC/C,MAAMD,EAAa,MAAMH,EAAgB,SAAS,IAAI,WAAW,CAAC,CAAC,CAAC,EAGpE,OAAOG,CAAU,EAAE,WAAQ,QAAK,IAAI,mCAAiCC,CAAS,CAAC,CAAC,CAClF,CAAC,CACH,CAAC,EAED,SAAS,qBAAsB,IAAM,CACnC,KAAK,8EAA+E,SAAY,CAE9F,MAAMpB,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGM2B,KAAc,QAClB,IAAI,yBAAuB,mBAAmB,CAChD,EACAxB,EAAqB,sBAAsBwB,CAAW,EACtD,MAAMN,EAAS,MAAMf,EAAU,QAAQ,CACrC,SAAU,MACV,aAAc,GAAG,GAAG,CACtB,CAAC,EAGD,OAAOe,CAAM,EAAE,QAAQM,CAAW,CACpC,CAAC,EAED,KAAK,sEAAuE,SAAY,CAEtF,MAAMZ,EAAQ,IAAI,MAAM,mBAAmB,EACrCT,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGAG,EAAqB,sBAAsBY,CAAK,EAChD,MAAMM,EAAS,MAAMf,EAAU,QAAQ,CACrC,SAAU,MACV,aAAc,GAAG,GAAG,CACtB,CAAC,EAGD,OAAOe,CAAM,EAAE,WAAQ,QAAK,IAAI,yBAAuBN,CAAK,CAAC,CAAC,CAChE,CAAC,CACH,CAAC,CACH,CAAC,EAED,SAAS,aAAc,IAAM,CAC3B,GAAG,yCAA0C,SAAY,CASvD,MAAMM,EAAS,MAPG,IAAI,iBACpB,GACAvB,EACAE,CACF,EAG+B,WAAW,CACxC,gBAAiB,CAAE,GAAI,YAAa,CACtC,CAAC,EAGD,OAAOF,EAAoB,gBAAgB,EAAE,qBAC3C,YACF,EACA,OAAOuB,CAAM,EAAE,WAAQ,SAAM,MAAS,CAAC,CACzC,CAAC,EAED,GAAG,qCAAsC,SAAY,CAEnD,MAAMf,EAAY,IAAI,iBACpB,GACAR,EACAE,CACF,EAGMe,EAAQ,IAAI,MAAM,mBAAmB,EAC3CX,EAAwB,sBAAsBW,CAAK,EACnD,MAAMM,EAAS,MAAMf,EAAU,WAAW,CACxC,gBAAiB,CAAE,GAAI,YAAa,CACtC,CAAC,EAGD,OAAOe,CAAM,EAAE,WAAQ,QAAK,IAAI,kBAAgBN,CAAK,CAAC,CAAC,CACzD,CAAC,CACH,CAAC,CACH,CAAC",
4
+ "sourcesContent": ["import {\n DeviceModelId,\n DisconnectError,\n type LoggerPublisherService,\n LogLevel,\n type LogParams,\n OpeningConnectionError,\n StaticDeviceModelDataSource,\n TransportConnectedDevice,\n type TransportDiscoveredDevice,\n} from \"@ledgerhq/device-management-kit\";\nimport { Left, Right } from \"purify-ts/Either\";\nimport { Subject } from \"rxjs\";\n\nimport { TRANSPORT_IDENTIFIER } from \"@api/transport/rnHidTransportIdentifier\";\n\nimport { HidTransportSendApduUnknownError } from \"./Errors\";\nimport { type NativeModuleWrapper } from \"./NativeModuleWrapper\";\nimport { RNHidTransport } from \"./RNHidTransport\";\nimport { type InternalDeviceDisconnected } from \"./types\";\n\nconst deviceModelDataSource = new StaticDeviceModelDataSource();\n\nconst makeMockTransportDiscoveredDevice = (\n id: string,\n deviceModelId: DeviceModelId,\n): TransportDiscoveredDevice => ({\n id,\n transport: TRANSPORT_IDENTIFIER,\n name: deviceModelId,\n deviceModel: deviceModelDataSource.getDeviceModel({\n id: deviceModelId,\n }),\n});\n\nconst mockDiscoveredDevice1 = makeMockTransportDiscoveredDevice(\n \"1\",\n DeviceModelId.NANO_S,\n);\nconst mockDiscoveredDevice2 = makeMockTransportDiscoveredDevice(\n \"2\",\n DeviceModelId.NANO_X,\n);\nconst mockDiscoveredDevice3 = makeMockTransportDiscoveredDevice(\n \"3\",\n DeviceModelId.NANO_S,\n);\n\ndescribe(\"RNHidTransport\", () => {\n let discoveredDevicesSubject: Subject<Array<TransportDiscoveredDevice>>;\n let deviceDisconnectedSubject: Subject<InternalDeviceDisconnected>;\n let transportLogsSubject: Subject<LogParams>;\n let nativeModuleWrapper: NativeModuleWrapper;\n const fakeLogger: LoggerPublisherService = {\n error: vi.fn(),\n warn: vi.fn(),\n info: vi.fn(),\n debug: vi.fn(),\n subscribers: [],\n };\n const loggerServiceFactory = vi.fn(() => fakeLogger);\n\n const wrapperStartScan = vi.fn();\n const wrapperStopScan = vi.fn();\n const wrapperConnectDevice = vi.fn();\n const wrapperDisconnectDevice = vi.fn();\n const wrapperSendApdu = vi.fn();\n\n beforeEach(() => {\n discoveredDevicesSubject = new Subject();\n deviceDisconnectedSubject = new Subject();\n transportLogsSubject = new Subject();\n\n vi.clearAllMocks();\n\n wrapperStartScan.mockResolvedValue(undefined);\n wrapperStopScan.mockResolvedValue(undefined);\n wrapperConnectDevice.mockResolvedValue(undefined);\n wrapperDisconnectDevice.mockResolvedValue(undefined);\n wrapperSendApdu.mockResolvedValue(undefined);\n\n nativeModuleWrapper = {\n startScan: wrapperStartScan,\n stopScan: wrapperStopScan,\n subscribeToDiscoveredDevicesEvents: vi.fn(() =>\n discoveredDevicesSubject.asObservable(),\n ),\n subscribeToDeviceDisconnectedEvents: vi.fn(() =>\n deviceDisconnectedSubject.asObservable(),\n ),\n subscribeToTransportLogs: vi.fn(() =>\n transportLogsSubject.asObservable(),\n ),\n connectDevice: wrapperConnectDevice,\n disconnectDevice: wrapperDisconnectDevice,\n sendApdu: wrapperSendApdu,\n };\n });\n\n test(\"getIdentifier returns TRANSPORT_IDENTIFIER\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transport.getIdentifier()).toBe(TRANSPORT_IDENTIFIER);\n });\n\n describe(\"isSupported returns the provided support flag\", () => {\n test(\"supported\", () => {\n // given\n const transportTrue = new RNHidTransport(\n true, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportTrue.isSupported()).toBe(true);\n });\n\n test(\"not supported\", () => {\n // given\n const transportFalse = new RNHidTransport(\n false, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportFalse.isSupported()).toBe(false);\n });\n });\n\n test(\"constructor subscribes to transport logs and calls logger methods\", () => {\n // given\n new RNHidTransport(true, nativeModuleWrapper, loggerServiceFactory);\n\n // when\n const logEvent: LogParams = [\n LogLevel.Info,\n \"Test message\",\n {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n },\n ];\n transportLogsSubject.next(logEvent);\n\n // then\n expect(fakeLogger.info).toHaveBeenCalledWith(\"Test message\", {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n });\n });\n\n describe(\"startDiscovering\", () => {\n it(\"calls startScan\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n transport.startDiscovering().subscribe();\n\n // then\n expect(wrapperStartScan).toHaveBeenCalled();\n });\n\n it(\"emits new discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const discoveredDevices: TransportDiscoveredDevice[] = [];\n\n // when\n transport.startDiscovering().subscribe({\n next: (device) => {\n discoveredDevices.push(device);\n if (discoveredDevices.length === 3) {\n try {\n // then\n expect(discoveredDevices).toEqual([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n error: reject,\n });\n // Simulate getObservableOfArraysNewItems behavior:\n // emit first array with one device then the subsequent arrays with a new device.\n discoveredDevicesSubject.next([mockDiscoveredDevice1]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.startDiscovering().subscribe({\n next: () => {},\n error: (err) => {\n try {\n // then\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n });\n\n describe(\"stopDiscovering\", () => {\n it(\"calls stopScan\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n await transport.stopDiscovering();\n\n // then\n expect(nativeModuleWrapper.stopScan).toHaveBeenCalled();\n });\n\n it(\"logs error when stopScan fails\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n\n // then\n await transport.stopDiscovering();\n expect(fakeLogger.error).toHaveBeenCalledWith(\n \"stopDiscovering error\",\n error,\n );\n });\n });\n\n describe(\"listenToKnownDevices\", () => {\n it(\"emits arrays of discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const observedEvents: Array<Array<TransportDiscoveredDevice>> = [];\n\n // when\n transport.listenToAvailableDevices().subscribe({\n next: (devices) => {\n observedEvents.push(devices);\n if (observedEvents.length === 2) {\n try {\n // then\n expect(observedEvents).toEqual([\n [mockDiscoveredDevice1, mockDiscoveredDevice2],\n [mockDiscoveredDevice1, mockDiscoveredDevice3],\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n complete: () => {\n reject(\"should not complete\");\n },\n error: reject,\n });\n\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"start scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.listenToAvailableDevices().subscribe({\n error: (err) => {\n // then\n try {\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n\n it(\"calls stopScan on unsubscribe\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop scan failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n const subscription = transport.listenToAvailableDevices().subscribe({});\n subscription.unsubscribe();\n\n expect(wrapperStopScan).toHaveBeenCalled();\n });\n });\n\n describe(\"connect\", () => {\n describe(\"connection successful\", () => {\n it(\"should return a Right(TransportConnectedDevice) on successful connection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result.isRight()).toBe(true);\n const connectedDevice = result.extract() as TransportConnectedDevice;\n expect(connectedDevice).toBeInstanceOf(TransportConnectedDevice);\n expect(connectedDevice.id).toBe(sessionId);\n expect(connectedDevice.deviceModel).toEqual(fakeDeviceModel);\n expect(connectedDevice.transport).toBe(TRANSPORT_IDENTIFIER);\n expect(connectedDevice.type).toBe(\"USB\");\n });\n\n test(\"should trigger onDisconnect when a matching disconnect event is emitted\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const onDisconnect = vi.fn();\n\n // when\n await transport.connect({\n deviceId: sessionId,\n onDisconnect,\n });\n deviceDisconnectedSubject.next({ sessionId });\n\n // then\n expect(onDisconnect).toHaveBeenCalledWith(sessionId);\n });\n\n test(\"should handle sendApdu success (Right)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Right(\"apduResponse\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu, false, 0);\n\n // then\n expect(nativeModuleWrapper.sendApdu).toHaveBeenCalledWith(\n sessionId,\n apdu,\n false,\n 0,\n );\n expect(apduResult).toEqual(Right(\"apduResponse\"));\n });\n\n test(\"should handle sendApdu failure (Left)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Left(\"some error\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu);\n\n // then\n expect(apduResult).toEqual(Left(\"some error\"));\n });\n\n test(\"should handle sendApdu rejection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n const apduError = new Error(\"apdu failed\");\n\n // when\n wrapperSendApdu.mockRejectedValueOnce(apduError);\n const apduResult = await connectedDevice.sendApdu(new Uint8Array([]));\n\n // then\n expect(apduResult).toEqual(\n Left(new HidTransportSendApduUnknownError(apduError)),\n );\n });\n });\n\n describe(\"connection failure\", () => {\n test(\"should return a Left when nativeModuleWrapper.connectDevice resolves a Left\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const errorResult = Left(\n new OpeningConnectionError(\"connection failed\"),\n );\n wrapperConnectDevice.mockResolvedValueOnce(errorResult);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(errorResult);\n });\n\n test(\"should return a Left when nativeModuleWrapper.connectDevice rejects\", async () => {\n // given\n const error = new Error(\"connection failed\");\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n wrapperConnectDevice.mockRejectedValueOnce(error);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(Left(new OpeningConnectionError(error)));\n });\n });\n });\n\n describe(\"disconnect\", () => {\n it(\"returns Right on successful disconnect\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.disconnect({\n connectedDevice: { id: \"session789\" } as TransportConnectedDevice,\n });\n\n // then\n expect(nativeModuleWrapper.disconnectDevice).toHaveBeenCalledWith(\n \"session789\",\n );\n expect(result).toEqual(Right(undefined));\n });\n\n it(\"returns Left on disconnect failure\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"disconnect failed\");\n wrapperDisconnectDevice.mockRejectedValueOnce(error);\n const result = await transport.disconnect({\n connectedDevice: { id: \"session000\" } as TransportConnectedDevice,\n });\n\n // then\n expect(result).toEqual(Left(new DisconnectError(error)));\n });\n });\n});\n"],
5
+ "mappings": "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",
6
6
  "names": ["import_device_management_kit", "import_Either", "import_rxjs", "import_rnHidTransportIdentifier", "import_Errors", "import_RNHidTransport", "deviceModelDataSource", "makeMockTransportDiscoveredDevice", "id", "deviceModelId", "mockDiscoveredDevice1", "mockDiscoveredDevice2", "mockDiscoveredDevice3", "discoveredDevicesSubject", "deviceDisconnectedSubject", "transportLogsSubject", "nativeModuleWrapper", "fakeLogger", "loggerServiceFactory", "wrapperStartScan", "wrapperStopScan", "wrapperConnectDevice", "wrapperDisconnectDevice", "wrapperSendApdu", "transport", "transportTrue", "transportFalse", "logEvent", "resolve", "reject", "discoveredDevices", "device", "e", "error", "err", "observedEvents", "devices", "fakeDeviceModel", "sessionId", "result", "connectedDevice", "onDisconnect", "apdu", "apduResult", "apduError", "errorResult"]
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/api/transport/RNHidTransport.test.ts"],
4
- "sourcesContent": ["import {\n DeviceModelId,\n DisconnectError,\n type LoggerPublisherService,\n LogLevel,\n type LogParams,\n OpeningConnectionError,\n StaticDeviceModelDataSource,\n TransportConnectedDevice,\n type TransportDiscoveredDevice,\n} from \"@ledgerhq/device-management-kit\";\nimport { Left, Right } from \"purify-ts/Either\";\nimport { Subject } from \"rxjs\";\n\nimport { TRANSPORT_IDENTIFIER } from \"@api/transport/rnHidTransportIdentifier\";\n\nimport { HidTransportSendApduUnknownError } from \"./Errors\";\nimport { type NativeModuleWrapper } from \"./NativeModuleWrapper\";\nimport { RNHidTransport } from \"./RNHidTransport\";\nimport { type InternalDeviceDisconnected } from \"./types\";\n\nconst deviceModelDataSource = new StaticDeviceModelDataSource();\n\nconst makeMockTransportDiscoveredDevice = (\n id: string,\n deviceModelId: DeviceModelId,\n): TransportDiscoveredDevice => ({\n id,\n transport: TRANSPORT_IDENTIFIER,\n name: deviceModelId,\n deviceModel: deviceModelDataSource.getDeviceModel({\n id: deviceModelId,\n }),\n});\n\nconst mockDiscoveredDevice1 = makeMockTransportDiscoveredDevice(\n \"1\",\n DeviceModelId.NANO_S,\n);\nconst mockDiscoveredDevice2 = makeMockTransportDiscoveredDevice(\n \"2\",\n DeviceModelId.NANO_X,\n);\nconst mockDiscoveredDevice3 = makeMockTransportDiscoveredDevice(\n \"3\",\n DeviceModelId.NANO_S,\n);\n\ndescribe(\"RNHidTransport\", () => {\n let discoveredDevicesSubject: Subject<Array<TransportDiscoveredDevice>>;\n let deviceDisconnectedSubject: Subject<InternalDeviceDisconnected>;\n let transportLogsSubject: Subject<LogParams>;\n let nativeModuleWrapper: NativeModuleWrapper;\n const fakeLogger: LoggerPublisherService = {\n error: vi.fn(),\n warn: vi.fn(),\n info: vi.fn(),\n debug: vi.fn(),\n subscribers: [],\n };\n const loggerServiceFactory = vi.fn(() => fakeLogger);\n\n const wrapperStartScan = vi.fn();\n const wrapperStopScan = vi.fn();\n const wrapperConnectDevice = vi.fn();\n const wrapperDisconnectDevice = vi.fn();\n const wrapperSendApdu = vi.fn();\n\n beforeEach(() => {\n discoveredDevicesSubject = new Subject();\n deviceDisconnectedSubject = new Subject();\n transportLogsSubject = new Subject();\n\n vi.clearAllMocks();\n\n wrapperStartScan.mockResolvedValue(undefined);\n wrapperStopScan.mockResolvedValue(undefined);\n wrapperConnectDevice.mockResolvedValue(undefined);\n wrapperDisconnectDevice.mockResolvedValue(undefined);\n wrapperSendApdu.mockResolvedValue(undefined);\n\n nativeModuleWrapper = {\n startScan: wrapperStartScan,\n stopScan: wrapperStopScan,\n subscribeToDiscoveredDevicesEvents: vi.fn(() =>\n discoveredDevicesSubject.asObservable(),\n ),\n subscribeToDeviceDisconnectedEvents: vi.fn(() =>\n deviceDisconnectedSubject.asObservable(),\n ),\n subscribeToTransportLogs: vi.fn(() =>\n transportLogsSubject.asObservable(),\n ),\n connectDevice: wrapperConnectDevice,\n disconnectDevice: wrapperDisconnectDevice,\n sendApdu: wrapperSendApdu,\n };\n });\n\n test(\"getIdentifier returns TRANSPORT_IDENTIFIER\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transport.getIdentifier()).toBe(TRANSPORT_IDENTIFIER);\n });\n\n describe(\"isSupported returns the provided support flag\", () => {\n test(\"supported\", () => {\n // given\n const transportTrue = new RNHidTransport(\n true, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportTrue.isSupported()).toBe(true);\n });\n\n test(\"not supported\", () => {\n // given\n const transportFalse = new RNHidTransport(\n false, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportFalse.isSupported()).toBe(false);\n });\n });\n\n test(\"constructor subscribes to transport logs and calls logger methods\", () => {\n // given\n new RNHidTransport(true, nativeModuleWrapper, loggerServiceFactory);\n\n // when\n const logEvent: LogParams = [\n LogLevel.Info,\n \"Test message\",\n {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n },\n ];\n transportLogsSubject.next(logEvent);\n\n // then\n expect(fakeLogger.info).toHaveBeenCalledWith(\"Test message\", {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n });\n });\n\n describe(\"startDiscovering\", () => {\n it(\"calls startScan\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n transport.startDiscovering().subscribe();\n\n // then\n expect(wrapperStartScan).toHaveBeenCalled();\n });\n\n it(\"emits new discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const discoveredDevices: TransportDiscoveredDevice[] = [];\n\n // when\n transport.startDiscovering().subscribe({\n next: (device) => {\n discoveredDevices.push(device);\n if (discoveredDevices.length === 3) {\n try {\n // then\n expect(discoveredDevices).toEqual([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n error: reject,\n });\n // Simulate getObservableOfArraysNewItems behavior:\n // emit first array with one device then the subsequent arrays with a new device.\n discoveredDevicesSubject.next([mockDiscoveredDevice1]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.startDiscovering().subscribe({\n next: () => {},\n error: (err) => {\n try {\n // then\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n });\n\n describe(\"stopDiscovering\", () => {\n it(\"calls stopScan\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n await transport.stopDiscovering();\n\n // then\n expect(nativeModuleWrapper.stopScan).toHaveBeenCalled();\n });\n\n it(\"logs error when stopScan fails\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n\n // then\n await transport.stopDiscovering();\n expect(fakeLogger.error).toHaveBeenCalledWith(\n \"stopDiscovering error\",\n error,\n );\n });\n });\n\n describe(\"listenToKnownDevices\", () => {\n it(\"emits arrays of discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const observedEvents: Array<Array<TransportDiscoveredDevice>> = [];\n\n // when\n transport.listenToAvailableDevices().subscribe({\n next: (devices) => {\n observedEvents.push(devices);\n if (observedEvents.length === 2) {\n try {\n // then\n expect(observedEvents).toEqual([\n [mockDiscoveredDevice1, mockDiscoveredDevice2],\n [mockDiscoveredDevice1, mockDiscoveredDevice3],\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n complete: () => {\n reject(\"should not complete\");\n },\n error: reject,\n });\n\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"start scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.listenToAvailableDevices().subscribe({\n error: (err) => {\n // then\n try {\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n\n it(\"calls stopScan on unsubscribe\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop scan failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n const subscription = transport.listenToAvailableDevices().subscribe({});\n subscription.unsubscribe();\n\n expect(wrapperStopScan).toHaveBeenCalled();\n });\n });\n\n describe(\"connect\", () => {\n describe(\"connection successful\", () => {\n it(\"should return a Right(TransportConnectedDevice) on successful connection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result.isRight()).toBe(true);\n const connectedDevice = result.extract() as TransportConnectedDevice;\n expect(connectedDevice).toBeInstanceOf(TransportConnectedDevice);\n expect(connectedDevice.id).toBe(sessionId);\n expect(connectedDevice.deviceModel).toEqual(fakeDeviceModel);\n expect(connectedDevice.transport).toBe(TRANSPORT_IDENTIFIER);\n expect(connectedDevice.type).toBe(\"USB\");\n });\n\n test(\"should trigger onDisconnect when a matching disconnect event is emitted\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const onDisconnect = vi.fn();\n\n // when\n await transport.connect({\n deviceId: sessionId,\n onDisconnect,\n });\n deviceDisconnectedSubject.next({ sessionId });\n\n // then\n expect(onDisconnect).toHaveBeenCalledWith(sessionId);\n });\n\n test(\"should handle sendApdu success (Right)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Right(\"apduResponse\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu, false, 0);\n\n // then\n expect(nativeModuleWrapper.sendApdu).toHaveBeenCalledWith(\n sessionId,\n apdu,\n false,\n 0,\n );\n expect(apduResult).toEqual(Right(\"apduResponse\"));\n });\n\n test(\"should handle sendApdu failure (Left)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Left(\"some error\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu);\n\n // then\n expect(apduResult).toEqual(Left(\"some error\"));\n });\n\n test(\"should handle sendApdu rejection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n const apduError = new Error(\"apdu failed\");\n\n // when\n wrapperSendApdu.mockRejectedValueOnce(apduError);\n const apduResult = await connectedDevice.sendApdu(new Uint8Array([]));\n\n // then\n expect(apduResult).toEqual(Left(new HidTransportSendApduUnknownError(apduError)));\n });\n });\n\n describe(\"connection failure\", () => {\n test(\"should return a Left when nativeModuleWrapper.connectDevice resolves a Left\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const errorResult = Left(\n new OpeningConnectionError(\"connection failed\"),\n );\n wrapperConnectDevice.mockResolvedValueOnce(errorResult);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(errorResult);\n });\n\n test(\"should return a Left when nativeModuleWrapper.connectDevice rejects\", async () => {\n // given\n const error = new Error(\"connection failed\");\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n wrapperConnectDevice.mockRejectedValueOnce(error);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(Left(new OpeningConnectionError(error)));\n });\n });\n });\n\n describe(\"disconnect\", () => {\n it(\"returns Right on successful disconnect\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.disconnect({\n connectedDevice: { id: \"session789\" } as TransportConnectedDevice,\n });\n\n // then\n expect(nativeModuleWrapper.disconnectDevice).toHaveBeenCalledWith(\n \"session789\",\n );\n expect(result).toEqual(Right(undefined));\n });\n\n it(\"returns Left on disconnect failure\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"disconnect failed\");\n wrapperDisconnectDevice.mockRejectedValueOnce(error);\n const result = await transport.disconnect({\n connectedDevice: { id: \"session000\" } as TransportConnectedDevice,\n });\n\n // then\n expect(result).toEqual(Left(new DisconnectError(error)));\n });\n });\n});\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import {\n DeviceModelId,\n DisconnectError,\n type LoggerPublisherService,\n LogLevel,\n type LogParams,\n OpeningConnectionError,\n StaticDeviceModelDataSource,\n TransportConnectedDevice,\n type TransportDiscoveredDevice,\n} from \"@ledgerhq/device-management-kit\";\nimport { Left, Right } from \"purify-ts/Either\";\nimport { Subject } from \"rxjs\";\n\nimport { TRANSPORT_IDENTIFIER } from \"@api/transport/rnHidTransportIdentifier\";\n\nimport { HidTransportSendApduUnknownError } from \"./Errors\";\nimport { type NativeModuleWrapper } from \"./NativeModuleWrapper\";\nimport { RNHidTransport } from \"./RNHidTransport\";\nimport { type InternalDeviceDisconnected } from \"./types\";\n\nconst deviceModelDataSource = new StaticDeviceModelDataSource();\n\nconst makeMockTransportDiscoveredDevice = (\n id: string,\n deviceModelId: DeviceModelId,\n): TransportDiscoveredDevice => ({\n id,\n transport: TRANSPORT_IDENTIFIER,\n name: deviceModelId,\n deviceModel: deviceModelDataSource.getDeviceModel({\n id: deviceModelId,\n }),\n});\n\nconst mockDiscoveredDevice1 = makeMockTransportDiscoveredDevice(\n \"1\",\n DeviceModelId.NANO_S,\n);\nconst mockDiscoveredDevice2 = makeMockTransportDiscoveredDevice(\n \"2\",\n DeviceModelId.NANO_X,\n);\nconst mockDiscoveredDevice3 = makeMockTransportDiscoveredDevice(\n \"3\",\n DeviceModelId.NANO_S,\n);\n\ndescribe(\"RNHidTransport\", () => {\n let discoveredDevicesSubject: Subject<Array<TransportDiscoveredDevice>>;\n let deviceDisconnectedSubject: Subject<InternalDeviceDisconnected>;\n let transportLogsSubject: Subject<LogParams>;\n let nativeModuleWrapper: NativeModuleWrapper;\n const fakeLogger: LoggerPublisherService = {\n error: vi.fn(),\n warn: vi.fn(),\n info: vi.fn(),\n debug: vi.fn(),\n subscribers: [],\n };\n const loggerServiceFactory = vi.fn(() => fakeLogger);\n\n const wrapperStartScan = vi.fn();\n const wrapperStopScan = vi.fn();\n const wrapperConnectDevice = vi.fn();\n const wrapperDisconnectDevice = vi.fn();\n const wrapperSendApdu = vi.fn();\n\n beforeEach(() => {\n discoveredDevicesSubject = new Subject();\n deviceDisconnectedSubject = new Subject();\n transportLogsSubject = new Subject();\n\n vi.clearAllMocks();\n\n wrapperStartScan.mockResolvedValue(undefined);\n wrapperStopScan.mockResolvedValue(undefined);\n wrapperConnectDevice.mockResolvedValue(undefined);\n wrapperDisconnectDevice.mockResolvedValue(undefined);\n wrapperSendApdu.mockResolvedValue(undefined);\n\n nativeModuleWrapper = {\n startScan: wrapperStartScan,\n stopScan: wrapperStopScan,\n subscribeToDiscoveredDevicesEvents: vi.fn(() =>\n discoveredDevicesSubject.asObservable(),\n ),\n subscribeToDeviceDisconnectedEvents: vi.fn(() =>\n deviceDisconnectedSubject.asObservable(),\n ),\n subscribeToTransportLogs: vi.fn(() =>\n transportLogsSubject.asObservable(),\n ),\n connectDevice: wrapperConnectDevice,\n disconnectDevice: wrapperDisconnectDevice,\n sendApdu: wrapperSendApdu,\n };\n });\n\n test(\"getIdentifier returns TRANSPORT_IDENTIFIER\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transport.getIdentifier()).toBe(TRANSPORT_IDENTIFIER);\n });\n\n describe(\"isSupported returns the provided support flag\", () => {\n test(\"supported\", () => {\n // given\n const transportTrue = new RNHidTransport(\n true, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportTrue.isSupported()).toBe(true);\n });\n\n test(\"not supported\", () => {\n // given\n const transportFalse = new RNHidTransport(\n false, // when\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n // then\n expect(transportFalse.isSupported()).toBe(false);\n });\n });\n\n test(\"constructor subscribes to transport logs and calls logger methods\", () => {\n // given\n new RNHidTransport(true, nativeModuleWrapper, loggerServiceFactory);\n\n // when\n const logEvent: LogParams = [\n LogLevel.Info,\n \"Test message\",\n {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n },\n ];\n transportLogsSubject.next(logEvent);\n\n // then\n expect(fakeLogger.info).toHaveBeenCalledWith(\"Test message\", {\n tag: \"TestTag\",\n data: { key: \"value\" },\n timestamp: 123456789,\n });\n });\n\n describe(\"startDiscovering\", () => {\n it(\"calls startScan\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n transport.startDiscovering().subscribe();\n\n // then\n expect(wrapperStartScan).toHaveBeenCalled();\n });\n\n it(\"emits new discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const discoveredDevices: TransportDiscoveredDevice[] = [];\n\n // when\n transport.startDiscovering().subscribe({\n next: (device) => {\n discoveredDevices.push(device);\n if (discoveredDevices.length === 3) {\n try {\n // then\n expect(discoveredDevices).toEqual([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n error: reject,\n });\n // Simulate getObservableOfArraysNewItems behavior:\n // emit first array with one device then the subsequent arrays with a new device.\n discoveredDevicesSubject.next([mockDiscoveredDevice1]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.startDiscovering().subscribe({\n next: () => {},\n error: (err) => {\n try {\n // then\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n });\n\n describe(\"stopDiscovering\", () => {\n it(\"calls stopScan\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n await transport.stopDiscovering();\n\n // then\n expect(nativeModuleWrapper.stopScan).toHaveBeenCalled();\n });\n\n it(\"logs error when stopScan fails\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n\n // then\n await transport.stopDiscovering();\n expect(fakeLogger.error).toHaveBeenCalledWith(\n \"stopDiscovering error\",\n error,\n );\n });\n });\n\n describe(\"listenToKnownDevices\", () => {\n it(\"emits arrays of discovered devices\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const observedEvents: Array<Array<TransportDiscoveredDevice>> = [];\n\n // when\n transport.listenToAvailableDevices().subscribe({\n next: (devices) => {\n observedEvents.push(devices);\n if (observedEvents.length === 2) {\n try {\n // then\n expect(observedEvents).toEqual([\n [mockDiscoveredDevice1, mockDiscoveredDevice2],\n [mockDiscoveredDevice1, mockDiscoveredDevice3],\n ]);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n },\n complete: () => {\n reject(\"should not complete\");\n },\n error: reject,\n });\n\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice2,\n ]);\n discoveredDevicesSubject.next([\n mockDiscoveredDevice1,\n mockDiscoveredDevice3,\n ]);\n });\n });\n\n it(\"propagates startScan error\", () => {\n return new Promise<void>((resolve, reject) => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"start scan failed\");\n wrapperStartScan.mockRejectedValueOnce(error);\n transport.listenToAvailableDevices().subscribe({\n error: (err) => {\n // then\n try {\n expect(err).toBe(error);\n resolve();\n } catch (e) {\n reject(e);\n }\n },\n });\n });\n });\n\n it(\"calls stopScan on unsubscribe\", () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"stop scan failed\");\n wrapperStopScan.mockRejectedValueOnce(error);\n const subscription = transport.listenToAvailableDevices().subscribe({});\n subscription.unsubscribe();\n\n expect(wrapperStopScan).toHaveBeenCalled();\n });\n });\n\n describe(\"connect\", () => {\n describe(\"connection successful\", () => {\n it(\"should return a Right(TransportConnectedDevice) on successful connection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result.isRight()).toBe(true);\n const connectedDevice = result.extract() as TransportConnectedDevice;\n expect(connectedDevice).toBeInstanceOf(TransportConnectedDevice);\n expect(connectedDevice.id).toBe(sessionId);\n expect(connectedDevice.deviceModel).toEqual(fakeDeviceModel);\n expect(connectedDevice.transport).toBe(TRANSPORT_IDENTIFIER);\n expect(connectedDevice.type).toBe(\"USB\");\n });\n\n test(\"should trigger onDisconnect when a matching disconnect event is emitted\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const onDisconnect = vi.fn();\n\n // when\n await transport.connect({\n deviceId: sessionId,\n onDisconnect,\n });\n deviceDisconnectedSubject.next({ sessionId });\n\n // then\n expect(onDisconnect).toHaveBeenCalledWith(sessionId);\n });\n\n test(\"should handle sendApdu success (Right)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Right(\"apduResponse\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu, false, 0);\n\n // then\n expect(nativeModuleWrapper.sendApdu).toHaveBeenCalledWith(\n sessionId,\n apdu,\n false,\n 0,\n );\n expect(apduResult).toEqual(Right(\"apduResponse\"));\n });\n\n test(\"should handle sendApdu failure (Left)\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n\n // when\n wrapperSendApdu.mockResolvedValueOnce(Left(\"some error\"));\n const apdu = new Uint8Array([1, 2, 3]);\n const apduResult = await connectedDevice.sendApdu(apdu);\n\n // then\n expect(apduResult).toEqual(Left(\"some error\"));\n });\n\n test(\"should handle sendApdu rejection\", async () => {\n // given\n const fakeDeviceModel = { model: \"TestModel\" };\n const sessionId = \"session123\";\n wrapperConnectDevice.mockResolvedValueOnce(\n Right({ sessionId, transportDeviceModel: fakeDeviceModel }),\n );\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n const result = await transport.connect({\n deviceId: sessionId,\n onDisconnect: vi.fn(),\n });\n const connectedDevice = result.extract() as TransportConnectedDevice;\n const apduError = new Error(\"apdu failed\");\n\n // when\n wrapperSendApdu.mockRejectedValueOnce(apduError);\n const apduResult = await connectedDevice.sendApdu(new Uint8Array([]));\n\n // then\n expect(apduResult).toEqual(\n Left(new HidTransportSendApduUnknownError(apduError)),\n );\n });\n });\n\n describe(\"connection failure\", () => {\n test(\"should return a Left when nativeModuleWrapper.connectDevice resolves a Left\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const errorResult = Left(\n new OpeningConnectionError(\"connection failed\"),\n );\n wrapperConnectDevice.mockResolvedValueOnce(errorResult);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(errorResult);\n });\n\n test(\"should return a Left when nativeModuleWrapper.connectDevice rejects\", async () => {\n // given\n const error = new Error(\"connection failed\");\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n wrapperConnectDevice.mockRejectedValueOnce(error);\n const result = await transport.connect({\n deviceId: \"any\",\n onDisconnect: vi.fn(),\n });\n\n // then\n expect(result).toEqual(Left(new OpeningConnectionError(error)));\n });\n });\n });\n\n describe(\"disconnect\", () => {\n it(\"returns Right on successful disconnect\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const result = await transport.disconnect({\n connectedDevice: { id: \"session789\" } as TransportConnectedDevice,\n });\n\n // then\n expect(nativeModuleWrapper.disconnectDevice).toHaveBeenCalledWith(\n \"session789\",\n );\n expect(result).toEqual(Right(undefined));\n });\n\n it(\"returns Left on disconnect failure\", async () => {\n // given\n const transport = new RNHidTransport(\n true,\n nativeModuleWrapper,\n loggerServiceFactory,\n );\n\n // when\n const error = new Error(\"disconnect failed\");\n wrapperDisconnectDevice.mockRejectedValueOnce(error);\n const result = await transport.disconnect({\n connectedDevice: { id: \"session000\" } as TransportConnectedDevice,\n });\n\n // then\n expect(result).toEqual(Left(new DisconnectError(error)));\n });\n });\n});\n"],
5
+ "mappings": "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",
6
6
  "names": ["DeviceModelId", "DisconnectError", "LogLevel", "OpeningConnectionError", "StaticDeviceModelDataSource", "TransportConnectedDevice", "Left", "Right", "Subject", "TRANSPORT_IDENTIFIER", "HidTransportSendApduUnknownError", "RNHidTransport", "deviceModelDataSource", "makeMockTransportDiscoveredDevice", "id", "deviceModelId", "mockDiscoveredDevice1", "mockDiscoveredDevice2", "mockDiscoveredDevice3", "discoveredDevicesSubject", "deviceDisconnectedSubject", "transportLogsSubject", "nativeModuleWrapper", "fakeLogger", "loggerServiceFactory", "wrapperStartScan", "wrapperStopScan", "wrapperConnectDevice", "wrapperDisconnectDevice", "wrapperSendApdu", "transport", "transportTrue", "transportFalse", "logEvent", "resolve", "reject", "discoveredDevices", "device", "e", "error", "err", "observedEvents", "devices", "fakeDeviceModel", "sessionId", "result", "connectedDevice", "onDisconnect", "apdu", "apduResult", "apduError", "errorResult"]
7
7
  }