mirror of
https://github.com/meshtastic/Meshtastic-Android.git
synced 2026-04-20 22:23:37 +00:00
184 lines
6.8 KiB
Kotlin
184 lines
6.8 KiB
Kotlin
/*
|
|
* Copyright (c) 2025 Meshtastic LLC
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
package com.geeksville.mesh.service
|
|
|
|
import co.touchlab.kermit.Logger
|
|
import com.geeksville.mesh.concurrent.handledLaunch
|
|
import kotlinx.coroutines.CoroutineScope
|
|
import kotlinx.coroutines.Dispatchers
|
|
import kotlinx.coroutines.SupervisorJob
|
|
import kotlinx.coroutines.delay
|
|
import org.meshtastic.core.analytics.platform.PlatformAnalytics
|
|
import org.meshtastic.core.data.repository.NodeRepository
|
|
import org.meshtastic.core.data.repository.RadioConfigRepository
|
|
import org.meshtastic.core.database.entity.MetadataEntity
|
|
import org.meshtastic.core.database.entity.MyNodeEntity
|
|
import org.meshtastic.core.service.ConnectionState
|
|
import org.meshtastic.proto.MeshProtos
|
|
import java.io.IOException
|
|
import javax.inject.Inject
|
|
import javax.inject.Singleton
|
|
|
|
@Suppress("LongParameterList")
|
|
@Singleton
|
|
class MeshConfigFlowManager
|
|
@Inject
|
|
constructor(
|
|
private val nodeManager: MeshNodeManager,
|
|
private val connectionManager: MeshConnectionManager,
|
|
private val nodeRepository: NodeRepository,
|
|
private val radioConfigRepository: RadioConfigRepository,
|
|
private val connectionStateHolder: ConnectionStateHandler,
|
|
private val serviceBroadcasts: MeshServiceBroadcasts,
|
|
private val analytics: PlatformAnalytics,
|
|
private val commandSender: MeshCommandSender,
|
|
private val packetHandler: PacketHandler,
|
|
) {
|
|
private var scope: CoroutineScope = CoroutineScope(Dispatchers.IO + SupervisorJob())
|
|
private val configOnlyNonce = 69420
|
|
private val nodeInfoNonce = 69421
|
|
private val wantConfigDelay = 100L
|
|
|
|
fun start(scope: CoroutineScope) {
|
|
this.scope = scope
|
|
}
|
|
|
|
private val newNodes = mutableListOf<MeshProtos.NodeInfo>()
|
|
val newNodeCount: Int
|
|
get() = newNodes.size
|
|
|
|
private var rawMyNodeInfo: MeshProtos.MyNodeInfo? = null
|
|
private var newMyNodeInfo: MyNodeEntity? = null
|
|
private var myNodeInfo: MyNodeEntity? = null
|
|
|
|
fun handleConfigComplete(configCompleteId: Int) {
|
|
when (configCompleteId) {
|
|
configOnlyNonce -> handleConfigOnlyComplete()
|
|
nodeInfoNonce -> handleNodeInfoComplete()
|
|
else -> Logger.w { "Config complete id mismatch: $configCompleteId" }
|
|
}
|
|
}
|
|
|
|
private fun handleConfigOnlyComplete() {
|
|
Logger.i { "Config-only complete" }
|
|
if (newMyNodeInfo == null) {
|
|
Logger.e { "Did not receive a valid config - newMyNodeInfo is null" }
|
|
} else {
|
|
myNodeInfo = newMyNodeInfo
|
|
Logger.i { "myNodeInfo committed successfully" }
|
|
}
|
|
|
|
scope.handledLaunch {
|
|
delay(wantConfigDelay)
|
|
sendHeartbeat()
|
|
delay(wantConfigDelay)
|
|
connectionManager.startNodeInfoOnly()
|
|
}
|
|
}
|
|
|
|
private fun sendHeartbeat() {
|
|
try {
|
|
packetHandler.sendToRadio(
|
|
MeshProtos.ToRadio.newBuilder().apply { heartbeat = MeshProtos.Heartbeat.getDefaultInstance() },
|
|
)
|
|
Logger.d { "Heartbeat sent between nonce stages" }
|
|
} catch (ex: IOException) {
|
|
Logger.w(ex) { "Failed to send heartbeat; proceeding with node-info stage" }
|
|
}
|
|
}
|
|
|
|
private fun handleNodeInfoComplete() {
|
|
Logger.i { "NodeInfo complete" }
|
|
val entities =
|
|
newNodes.map { info ->
|
|
nodeManager.installNodeInfo(info, withBroadcast = false)
|
|
nodeManager.nodeDBbyNodeNum[info.num]!!
|
|
}
|
|
newNodes.clear()
|
|
|
|
scope.handledLaunch {
|
|
myNodeInfo?.let {
|
|
nodeRepository.installConfig(it, entities)
|
|
sendAnalytics(it)
|
|
}
|
|
nodeManager.isNodeDbReady.value = true
|
|
nodeManager.allowNodeDbWrites.value = true
|
|
connectionStateHolder.setState(ConnectionState.Connected)
|
|
serviceBroadcasts.broadcastConnection()
|
|
connectionManager.onHasSettings()
|
|
}
|
|
}
|
|
|
|
private fun sendAnalytics(mi: MyNodeEntity) {
|
|
analytics.setDeviceAttributes(mi.firmwareVersion ?: "unknown", mi.model ?: "unknown")
|
|
}
|
|
|
|
fun handleMyInfo(myInfo: MeshProtos.MyNodeInfo) {
|
|
Logger.i { "MyNodeInfo received: ${myInfo.myNodeNum}" }
|
|
rawMyNodeInfo = myInfo
|
|
nodeManager.myNodeNum = myInfo.myNodeNum
|
|
regenMyNodeInfo()
|
|
|
|
scope.handledLaunch {
|
|
radioConfigRepository.clearChannelSet()
|
|
radioConfigRepository.clearLocalConfig()
|
|
radioConfigRepository.clearLocalModuleConfig()
|
|
}
|
|
}
|
|
|
|
fun handleLocalMetadata(metadata: MeshProtos.DeviceMetadata) {
|
|
Logger.i { "Local Metadata received" }
|
|
regenMyNodeInfo(metadata)
|
|
}
|
|
|
|
fun handleNodeInfo(info: MeshProtos.NodeInfo) {
|
|
newNodes.add(info)
|
|
}
|
|
|
|
private fun regenMyNodeInfo(metadata: MeshProtos.DeviceMetadata? = MeshProtos.DeviceMetadata.getDefaultInstance()) {
|
|
val myInfo = rawMyNodeInfo
|
|
if (myInfo != null) {
|
|
val mi =
|
|
with(myInfo) {
|
|
MyNodeEntity(
|
|
myNodeNum = myNodeNum,
|
|
model =
|
|
when (val hwModel = metadata?.hwModel) {
|
|
null,
|
|
MeshProtos.HardwareModel.UNSET,
|
|
-> null
|
|
else -> hwModel.name.replace('_', '-').replace('p', '.').lowercase()
|
|
},
|
|
firmwareVersion = metadata?.firmwareVersion,
|
|
couldUpdate = false,
|
|
shouldUpdate = false,
|
|
currentPacketId = commandSender.getCurrentPacketId() and 0xffffffffL,
|
|
messageTimeoutMsec = 300000,
|
|
minAppVersion = minAppVersion,
|
|
maxChannels = 8,
|
|
hasWifi = metadata?.hasWifi == true,
|
|
deviceId = deviceId.toStringUtf8(),
|
|
)
|
|
}
|
|
if (metadata != null && metadata != MeshProtos.DeviceMetadata.getDefaultInstance()) {
|
|
scope.handledLaunch { nodeRepository.insertMetadata(MetadataEntity(mi.myNodeNum, metadata)) }
|
|
}
|
|
newMyNodeInfo = mi
|
|
}
|
|
}
|
|
}
|