Завершение сквозного объекта завершено, не вызывается

#swift #swiftui #core-bluetooth #combine #publisher

#swift #swiftui #ядро-bluetooth #объединить #издатель

Вопрос:

Я пытаюсь связаться с CoreBluetooth с помощью Combine, но мой обработчик завершения a PassthroughSubject не вызывается. Ниже вы можете увидеть примерный макет кода. DetailViewModel Содержит периферийное устройство Bluetooth и данные для отправки.

 final class DetailViewModel: NSObject, ObservableObject, CBPeripheralDelegate {
    // Called when the correct write characteristic is found
    private var writeCharacteristicReceived = PassthroughSubject<CBCharacteristic, Never>()
    // Used to send and listen for peripheral data
    private var bluetoothDidChange = PassthroughSubject<Data, Error>()

    func open() -> AnyPublisher<Data, Error> {
        writeCharacteristicReceived.tryMap { characteristic -> AnyPublisher<Data, Error> in
            print("Write char", characteristic)

            let data: Data = try constructPayload()

            self.peripheral?.writeValue(data, for: characteristic, type: .withoutResponse)

            return self.bluetoothDidChange.eraseToAnyPublisher()
        }
        .switchToLatest()
        .eraseToAnyPublisher()
    }

    func peripheral(_ peripheral: CBPeripheral, didUpdateValueFor characteristic: CBCharacteristic, error: Error?) {
        print("Did update value", characteristic.value ?? Data())
        guard let value = characteristic.value, value.count >= 84 else { return }

        defer {
            // Never called
            bluetoothDidChange.send(completion: .finished)
        }

        do {
            let message: Data = try parse(value)
            bluetoothDidChange.send(message)
            // Never called when placed here either
            // bluetoothDidChange.send(completion: .finished)
        } catch {
            bluetoothDidChange.send(completion: .failure(error))
        }
    }
}
 

Затем я прослушиваю эти изменения в самом представлении следующим образом

 viewModel.open().sink(receiveCompletion: { (completion) in
    print("Open completion: (completion)")
}, receiveValue: { (payload) in
    print("Open payload (payload)")
}).store(in: amp;cancellable)
 

Теперь это отлично работает для получения значений время от времени, и блок завершения вызывается правильно при возникновении ошибки. Но я никогда не получаю завершенный обработчик завершения, даже когда я специально это делаю send(completion: .finished) . Кто-нибудь может мне помочь?

Ответ №1:

Проблема, связанная с writeCharacteristicReceived.send(completion: .finished) тем, что никогда не вызывается. Добавление этого к функции, вызываемой делегатом, решило проблему.

 func peripheral(_ peripheral: CBPeripheral, didDiscoverDescriptorsFor characteristic: CBCharacteristic, error: Error?) {
    if let error = error {
        writeCharacteristicDiscovered.send(completion: .failure(error))
    }
    writeCharacteristicDiscovered.send(characteristic)
    writeCharacteristicDiscovered.send(completion: .finished)
}
 

Комментарии:

1. «… в правильном месте» , пожалуйста, завершите свой ответ, включив код, в котором это было необходимо, чтобы помочь другим, ищущим аналогичное решение.