You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

281 lines
8.8 KiB

package net.aiterp.git.ykonsole2.infrastructure
import com.welie.blessed.*
import com.welie.blessed.BluetoothGattCharacteristic.WriteType.WITHOUT_RESPONSE
import kotlinx.coroutines.*
import net.aiterp.git.ykonsole2.domain.models.Device
import net.aiterp.git.ykonsole2.domain.runtime.*
import net.aiterp.git.ykonsole2.infrastructure.drivers.abstracts.ActiveDriver
import net.aiterp.git.ykonsole2.infrastructure.iconsole.*
import net.aiterp.git.ykonsole2.application.logging.log
import java.time.Instant
import java.util.*
import kotlin.time.Duration
import kotlin.time.Duration.Companion.milliseconds
class IConsole : ActiveDriver() {
private val logger = log
private var lastTime = 0
private var lastLevel = 0
private var running = false
private var connected = false
private var tryingSince: Instant? = null
private var maxLevel = 20
private var lastCals = 0
private var lastMeters = 0
private val queue = Collections.synchronizedSet<Request?>(LinkedHashSet())
private var central: BluetoothCentralManager? = null
private var current: BluetoothPeripheral? = null
private var btCommandInput: BluetoothGattCharacteristic? = null
private fun onConnect(device: Device, output: FlowBus<Event>) {
tryingSince = Instant.now()
if (!device.connectionString.startsWith("iconsole:")) {
logger.info("Ignoring non-iConsole $device")
return
}
val connectionString = device.connectionString.substring("iconsole:".length)
val cbPeripheral = object : BluetoothPeripheralCallback() {
override fun onServicesDiscovered(peripheral: BluetoothPeripheral, services: MutableList<BluetoothGattService>) {
logger.info("Checking device...")
val service1 = services.firstOrNull { it.uuid == S1_SERVICE }
val service2 = services.firstOrNull { it.uuid == S2_SERVICE }
if (service1 != null && service2 != null) {
btCommandInput = service1.getCharacteristic(S1_CHAR_COMMAND_INPUT)
val s1out = service1.getCharacteristic(S1_CHAR_DATA_OUTPUT)
val s1mys = service1.getCharacteristic(S1_CHAR_MYSTERY_OUTPUT)
val s2mys = service2.getCharacteristic(S2_CHAR_MYSTERY_OUTPUT)
if (btCommandInput == null || s1out == null || s1mys == null || s2mys == null) {
logger.warn("Something is off about this i-CONSOLE device")
output.emitBlocking(ErrorOccurred("(Maybe) a malfunctioning i-CONSOLE device"))
peripheral.cancelConnection()
return
}
peripheral.setNotify(btCommandInput!!, true)
peripheral.setNotify(s1out, true)
peripheral.setNotify(s1mys, true)
peripheral.setNotify(s2mys, true)
logger.info("Device setup successfully; sending first request...")
runBlocking {
peripheral.writeCharacteristic(
btCommandInput!!,
AckRequest.toBytes(),
WITHOUT_RESPONSE,
)
}
current = peripheral
} else {
logger.warn("This is maybe not an i-CONSOLE device, since it lacks the required services")
output.emitBlocking(ErrorOccurred("(Maybe) not an i-CONSOLE device"))
peripheral.cancelConnection()
}
}
override fun onCharacteristicUpdate(
peripheral: BluetoothPeripheral,
value: ByteArray?,
characteristic: BluetoothGattCharacteristic,
status: BluetoothCommandStatus,
) {
val res = value?.let { Response.fromBytes(it) }
if (!connected && res is AckResponse) {
logger.info("Device is now ready")
connected = true
output.emitBlocking(Connected)
}
if (res is MaxLevelResponse) {
maxLevel = res.level
}
if (res is ControlStateResponse) {
if (res.value == 1) {
running = true
output.emitBlocking(Started)
} else {
running = false
output.emitBlocking(Stopped)
}
}
if (res is WorkoutStatusResponse) {
val currentTime = (res.minutes * 60) + res.seconds
if (currentTime > lastTime) {
lastTime = currentTime
lastCals = maxOf(res.calories, lastCals)
lastMeters = maxOf((res.distance * 1000).toInt(), lastMeters)
output.emitBlocking(ValuesReceived(listOf(
Time(lastTime),
Calories(lastCals),
Distance(lastMeters),
Level(res.level),
)))
}
}
}
}
val cbScan = object : BluetoothCentralManagerCallback() {
override fun onScanStarted() {
logger.info("Scanning for $connectionString...")
}
override fun onScanStopped() {
logger.info("Scan stopped")
tryingSince?.let { ts ->
if (ts.isBefore(Instant.now().minusSeconds(20))) {
central?.stopScan()
tryingSince = null
runBlocking {
output.emit(ErrorOccurred("Connection timeout after 20 seconds"))
output.emit(Disconnected)
}
}
}
}
override fun onDiscoveredPeripheral(peripheral: BluetoothPeripheral, scanResult: ScanResult) {
logger.info("Connecting to ${peripheral.name} (${peripheral.address})...")
tryingSince = null
central?.connectPeripheral(peripheral, cbPeripheral)
central?.stopScan()
}
override fun onDisconnectedPeripheral(peripheral: BluetoothPeripheral, status: BluetoothCommandStatus) {
if (peripheral.address == current?.address) {
current = null
output.emitBlocking(Disconnected)
}
}
}
central = BluetoothCentralManager(cbScan).apply {
scanForPeripheralsWithAddresses(arrayOf(connectionString))
}
}
private fun onDisconnect() {
lastTime = 0
lastCals = 0
lastMeters = 0
running = false
connected = false
current?.cancelConnection()
}
private fun onSetValue(value: Value) {
if (current == null || btCommandInput === null || !running) return
if (value is Level) {
lastLevel = value.toInt()
queue += SetResistanceLevelRequest(minOf(lastLevel, maxLevel))
}
}
private fun onStart() {
if (current == null || btCommandInput === null || running) return
if (lastTime == 0) {
logger.info("STARTED")
queue += AckRequest
queue += SetWorkoutModeRequest(0)
queue += SetWorkoutParamsRequest()
queue += SetWorkoutControlStateRequest(1)
} else {
logger.info("RESUMED")
queue += SetWorkoutControlStateRequest(1)
}
}
private fun onStop() {
if (current == null || btCommandInput === null || !running) return
queue += SetWorkoutControlStateRequest(2)
}
override suspend fun onCommand(command: Command, output: FlowBus<Event>) {
when (command) {
is ConnectCommand -> onConnect(command.device, output)
DisconnectCommand -> onDisconnect()
is SetValueCommand -> onSetValue(command.value)
StartCommand -> onStart()
StopCommand -> onStop()
SkipCommand -> Unit
}
}
override suspend fun onTick(output: FlowBus<Event>) {
current?.apply {
if (connected && btCommandInput != null) {
val s1input = btCommandInput!!
// Fetch one queued event when running, otherwise take up to five at a time
val max = if (running) 1 else 5
for (i in 0 until max) {
if (queue.isEmpty()) {
break
}
val first = queue.first()
writeCharacteristic(s1input, first.toBytes(), WITHOUT_RESPONSE)
queue -= first
delay(pollDuration)
}
if (running) {
// Fetch current state
writeCharacteristic(
s1input,
GetWorkoutStateRequest.toBytes(),
WITHOUT_RESPONSE,
)
} else {
// This is just to keep the connection alive, as far as I know…
runBlocking {
writeCharacteristic(
s1input,
GetMaxLevelRequest.toBytes(),
WITHOUT_RESPONSE,
)
delay(pollDuration)
}
}
}
}
delay(pollDuration)
}
companion object {
private val S1_SERVICE = UUID.fromString("49535343-fe7d-4ae5-8fa9-9fafd205e455")
private val S1_CHAR_COMMAND_INPUT = UUID.fromString("49535343-8841-43f4-a8d4-ecbe34729bb3")
private val S1_CHAR_DATA_OUTPUT = UUID.fromString("49535343-1e4d-4bd9-ba61-23c647249616")
private val S1_CHAR_MYSTERY_OUTPUT = UUID.fromString("49535343-4c8a-39b3-2f49-511cff073b7e")
private val S2_SERVICE = UUID.fromString("49535343-5d82-6099-9348-7aac4d5fbc51")
private val S2_CHAR_MYSTERY_OUTPUT = UUID.fromString("49535343-026e-3a9b-954c-97daef17e26e")
private val pollDuration: Duration = 500.milliseconds
}
}