Skip to content

Commit d84159c

Browse files
committed
Merge remote-tracking branch 'benma/bt'
2 parents fc5ccb5 + c0f1601 commit d84159c

File tree

4 files changed

+101
-32
lines changed

4 files changed

+101
-32
lines changed

backend/devices/bluetooth/bluetooth.go

+44-6
Original file line numberDiff line numberDiff line change
@@ -15,25 +15,63 @@
1515
package bluetooth
1616

1717
import (
18+
"encoding/json"
19+
"fmt"
20+
1821
"github.com/BitBoxSwiss/bitbox-wallet-app/util/locker"
1922
"github.com/BitBoxSwiss/bitbox-wallet-app/util/observable"
2023
"github.com/BitBoxSwiss/bitbox-wallet-app/util/observable/action"
2124
"github.com/sirupsen/logrus"
2225
)
2326

27+
// ConnectionState represents the a peripheral's connection state.
28+
type ConnectionState string
29+
30+
const (
31+
// ConnectionStateDiscovered means the peripheral is discovered.
32+
ConnectionStateDiscovered ConnectionState = "discovered"
33+
// ConnectionStateConnecting means a connection & pairing attempt is in progress.
34+
ConnectionStateConnecting ConnectionState = "connecting"
35+
// ConnectionStateConnected means successfully connected and paired.
36+
ConnectionStateConnected ConnectionState = "connected"
37+
// ConnectionStateError means there was a connection error, see ConnectionError.
38+
ConnectionStateError ConnectionState = "error"
39+
)
40+
41+
// UnmarshalJSON verifies that only a valid state string can be deserialized.
42+
func (cs *ConnectionState) UnmarshalJSON(data []byte) error {
43+
var s string
44+
if err := json.Unmarshal(data, &s); err != nil {
45+
return err
46+
}
47+
switch ConnectionState(s) {
48+
case ConnectionStateDiscovered,
49+
ConnectionStateConnecting,
50+
ConnectionStateConnected,
51+
ConnectionStateError:
52+
*cs = ConnectionState(s)
53+
return nil
54+
default:
55+
return fmt.Errorf("invalid ConnectionState: %q", s)
56+
}
57+
}
58+
2459
// Peripheral is a bluetooth peripheral.
2560
type Peripheral struct {
26-
Identifier string `json:"identifier"`
27-
Name string `json:"name"`
28-
ConnectionError *string `json:"connectionError,omitempty"`
61+
Identifier string `json:"identifier"`
62+
Name string `json:"name"`
63+
// ConnectionState shows the current connection phase
64+
ConnectionState ConnectionState `json:"connectionState"`
65+
ConnectionError *string `json:"connectionError,omitempty"`
2966
}
3067

3168
// State contains everything needed to render bluetooth peripherals and other data in the frontend.
3269
type State struct {
3370
// BluetoothAvailable is false if bluetooth is powered off or otherwise unavailable.
34-
BluetoothAvailable bool `json:"bluetoothAvailable"`
35-
Peripherals []*Peripheral `json:"peripherals"`
36-
Connecting bool `json:"connecting"`
71+
BluetoothAvailable bool `json:"bluetoothAvailable"`
72+
// Scanning is true if we are currently scanning for peripherals.
73+
Scanning bool `json:"scanning"`
74+
Peripherals []*Peripheral `json:"peripherals"`
3775
}
3876

3977
// Bluetooth manages a list of peripherals.

frontends/ios/BitBoxApp/BitBoxApp/Bluetooth.swift

+43-19
Original file line numberDiff line numberDiff line change
@@ -19,15 +19,25 @@ struct ProductInfo: Codable {
1919
}
2020
}
2121

22+
enum ConnectionState: String, Codable {
23+
case discovered
24+
// from the moment we try to connect until after we are paired (or until either step fails).
25+
case connecting
26+
case connected
27+
// something went wrong, see connectionError.
28+
case error
29+
}
30+
2231
struct State {
2332
var bluetoothAvailable: Bool
33+
var scanning: Bool
2434
var discoveredPeripherals: [UUID: PeripheralMetadata]
25-
var connecting: Bool
2635
}
2736

2837
struct PeripheralMetadata {
2938
let peripheral: CBPeripheral
3039
let discoveredDate: Date
40+
var connectionState: ConnectionState
3141
var connectionError: String? = nil
3242
}
3343

@@ -42,14 +52,18 @@ var pairedDeviceIdentifiers: Set<String> {
4252
}
4353

4454
class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CBPeripheralDelegate {
45-
private var state: State = State(bluetoothAvailable: false, discoveredPeripherals: [:], connecting: false)
55+
private var state: State = State(
56+
bluetoothAvailable: false,
57+
scanning: false,
58+
discoveredPeripherals: [:]
59+
)
4660

4761
var centralManager: CBCentralManager!
4862
var connectedPeripheral: CBPeripheral?
4963
var pWriter: CBCharacteristic?
5064
var pReader: CBCharacteristic?
5165
var pProduct: CBCharacteristic?
52-
66+
5367
private var isPaired: Bool = false
5468

5569
// Peripherals in this set will not be auto-connected even if previously paired.
@@ -72,10 +86,12 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
7286
}
7387

7488
func connect(to peripheralID: UUID) {
75-
guard let metadata = state.discoveredPeripherals[peripheralID] else { return }
89+
guard var metadata = state.discoveredPeripherals[peripheralID] else { return }
7690
centralManager.stopScan()
77-
state.discoveredPeripherals[peripheralID]?.connectionError = nil
78-
state.connecting = true
91+
metadata.connectionError = nil
92+
metadata.connectionState = .connecting
93+
state.discoveredPeripherals[peripheralID] = metadata
94+
state.scanning = false
7995
updateBackendState()
8096
centralManager.connect(metadata.peripheral, options: nil)
8197
}
@@ -85,6 +101,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
85101
!centralManager.isScanning,
86102
connectedPeripheral == nil else { return }
87103
state.discoveredPeripherals.removeAll()
104+
state.scanning = true
88105
updateBackendState()
89106
centralManager.scanForPeripherals(
90107
withServices: [CBUUID(string: "e1511a45-f3db-44c0-82b8-6c880790d1f1")],
@@ -113,7 +130,8 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
113130
if state.discoveredPeripherals[identifier] == nil {
114131
state.discoveredPeripherals[identifier] = PeripheralMetadata(
115132
peripheral: peripheral,
116-
discoveredDate: Date()
133+
discoveredDate: Date(),
134+
connectionState: .discovered
117135
)
118136
print("BLE: discovered \(peripheral.name ?? "unknown device")")
119137
if let data = advertisementData["kCBAdvDataManufacturerData"] as? Data {
@@ -140,8 +158,6 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
140158
print("BLE: Connected to \(peripheral.name ?? "unknown device")")
141159

142160
state.discoveredPeripherals[peripheral.identifier]?.connectionError = nil
143-
state.connecting = false
144-
updateBackendState()
145161

146162
connectedPeripheral = peripheral
147163
peripheral.delegate = self
@@ -150,11 +166,12 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
150166

151167
func centralManager(_ central: CBCentralManager, didFailToConnect peripheral: CBPeripheral, error: Error?) {
152168
let errorMessage = error?.localizedDescription ?? "unknown error"
169+
state.discoveredPeripherals[peripheral.identifier]?.connectionState = .error
153170
state.discoveredPeripherals[peripheral.identifier]?.connectionError = errorMessage
154-
state.connecting = false
155-
dontAutoConnectSet.insert(peripheral.identifier)
156171
updateBackendState()
172+
dontAutoConnectSet.insert(peripheral.identifier)
157173
print("BLE: connection failed to \(peripheral.name ?? "unknown device"): \(errorMessage)")
174+
restartScan()
158175
}
159176

160177
func peripheral(_ peripheral: CBPeripheral, didDiscoverServices error: Error?) {
@@ -241,27 +258,28 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
241258
// Add to paired devices
242259
pairedDeviceIdentifiers.insert(peripheral.identifier.uuidString)
243260
}
261+
state.discoveredPeripherals[peripheral.identifier]?.connectionState = .connected
262+
updateBackendState()
244263
// Invoke device manager to scan now, which will make it detect the device being connected
245264
// (or disconnected, in case the product string indicates that) now instead of waiting for
246265
// the next scan.
247266
MobileserverUsbUpdate()
248267
}
249268
}
250-
269+
251270
func handleDisconnect() {
252271
connectedPeripheral = nil
253272
pReader = nil
254273
pWriter = nil
255274
pProduct = nil
256275
state.discoveredPeripherals.removeAll()
257-
state.connecting = false
258-
isPaired = false
276+
isPaired = false
259277
updateBackendState()
260-
278+
261279
// Have the backend scan right away, which will make it detect that we disconnected.
262280
// Otherwise there would be up to a second of delay (the backend device manager scan interval).
263281
MobileserverUsbUpdate()
264-
282+
265283
restartScan()
266284
}
267285

@@ -298,7 +316,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
298316
let value = pProduct.value else {
299317
return nil
300318
}
301-
319+
302320
if value.isEmpty {
303321
return nil
304322
}
@@ -319,25 +337,31 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
319337
struct PeripheralJSON: Codable {
320338
let identifier: String
321339
let name: String
340+
let connectionState: ConnectionState
322341
let connectionError: String?
323342
}
324343

325344
struct StateJSON: Codable {
326345
let bluetoothAvailable: Bool
346+
let scanning: Bool
327347
let peripherals: [PeripheralJSON]
328-
let connecting: Bool
329348
}
330349

331350
// Convert discoveredPeripherals to the JSON structure
332351
let peripherals = Array(state.discoveredPeripherals.values).map { metadata in
333352
PeripheralJSON(
334353
identifier: metadata.peripheral.identifier.uuidString,
335354
name: metadata.peripheral.name ?? "BitBox",
355+
connectionState: metadata.connectionState,
336356
connectionError: metadata.connectionError
337357
)
338358
}
339359

340-
let state = StateJSON(bluetoothAvailable: state.bluetoothAvailable, peripherals: peripherals, connecting: state.connecting)
360+
let state = StateJSON(
361+
bluetoothAvailable: state.bluetoothAvailable,
362+
scanning: state.scanning,
363+
peripherals: peripherals
364+
)
341365

342366
do {
343367
let encoder = JSONEncoder()

frontends/web/src/api/bluetooth.ts

+10-3
Original file line numberDiff line numberDiff line change
@@ -20,13 +20,20 @@ import { subscribeEndpoint, TUnsubscribe } from './subscribe';
2020
export type TPeripheral = {
2121
identifier: string;
2222
name: string;
23-
connectionError?: string;
24-
};
23+
} & (
24+
| {
25+
connectionState: 'discovered' | 'connecting' | 'connected';
26+
}
27+
| {
28+
connectionState: 'error';
29+
connectionError: string;
30+
}
31+
);
2532

2633
export type TState = {
2734
bluetoothAvailable: boolean;
35+
scanning: boolean;
2836
peripherals: TPeripheral[];
29-
connecting: boolean;
3037
};
3138

3239
export const getState = (): Promise<TState> => {

frontends/web/src/components/bluetooth/bluetooth.tsx

+4-4
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ const _Bluetooth = () => {
3737
{t('bluetooth.select')}
3838
</div>
3939
<div className={styles.container}>
40-
{ state.connecting ? <p>connecting</p> : null }
40+
{ state.scanning ? 'scanning' : null }
4141
{state.peripherals.map(peripheral => {
4242
return (
4343
<ActionableItem
@@ -46,14 +46,14 @@ const _Bluetooth = () => {
4646
<span>
4747
{ peripheral.name !== '' ? peripheral.name : peripheral.identifier }
4848
{' '}
49-
{ peripheral.connectionError ? (
49+
{ peripheral.connectionState === 'error' ? (
5050
<Badge type="danger">
5151
{t('bluetooth.connectionFailed')}
5252
</Badge>
5353
) : null }
54-
{ peripheral.connectionError ? (
54+
{ peripheral.connectionState === 'error' ? (
5555
<p>{ peripheral.connectionError }</p>
56-
) : null }
56+
) : peripheral.connectionState }
5757
</span>
5858

5959
</ActionableItem>

0 commit comments

Comments
 (0)