All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.fr.acinq.lightning.blockchain.electrum.ElectrumWatcher.kt Maven / Gradle / Ivy

There is a newer version: 1.8.4
Show newest version
package fr.acinq.lightning.blockchain.electrum

import fr.acinq.bitcoin.ByteVector32
import fr.acinq.bitcoin.Transaction
import fr.acinq.lightning.blockchain.*
import fr.acinq.lightning.transactions.Scripts
import fr.acinq.lightning.utils.currentTimestampMillis
import kotlinx.coroutines.*
import kotlinx.coroutines.channels.BufferOverflow
import kotlinx.coroutines.channels.Channel
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.asSharedFlow
import kotlinx.coroutines.flow.consumeAsFlow
import org.kodein.log.LoggerFactory
import org.kodein.log.newLogger
import kotlin.math.max

class ElectrumWatcher(val client: IElectrumClient, val scope: CoroutineScope, loggerFactory: LoggerFactory) : CoroutineScope by scope {

    private val logger = loggerFactory.newLogger(this::class)
    private val mailbox = Channel(Channel.BUFFERED)

    private val _notificationsFlow = MutableSharedFlow(replay = 0, extraBufferCapacity = 64, onBufferOverflow = BufferOverflow.SUSPEND)
    fun openWatchNotificationsFlow(): Flow = _notificationsFlow.asSharedFlow()

    // this is used by a Swift watch-tower module in the Phoenix iOS app to tell when the watcher is up-to-date
    // the value that is emitted in the time elapsed (in milliseconds) since the watcher is ready and idle
    private val _uptodateFlow = MutableSharedFlow(replay = 0, extraBufferCapacity = 64, onBufferOverflow = BufferOverflow.SUSPEND)
    fun openUpToDateFlow(): Flow = _uptodateFlow.asSharedFlow()

    suspend fun watch(watch: Watch) {
        mailbox.send(WatcherCommand.AddWatch(watch))
    }

    suspend fun publish(tx: Transaction) {
        mailbox.send(WatcherCommand.Publish(tx))
    }

    private sealed interface WatcherCommand {
        data class AddWatch(val watch: Watch) : WatcherCommand
        data class ProcessNotification(val notification: ElectrumResponse) : WatcherCommand
        data class ProcessConnectionStatus(val status: ElectrumConnectionStatus) : WatcherCommand
        data class Publish(val tx: Transaction) : WatcherCommand
        object NotifyIfReady : WatcherCommand
    }

    private data class State(
        val height: Int, // current block height. 0 means that we're not connected
        val watches: Set = setOf(),
        val scriptHashStatus: Map = mapOf(),
        val scriptHashSubscriptions: Set = setOf(),
        val publishQueue: Set = setOf(),
        val block2tx: Map> = mapOf(),
        val sent: Set = setOf(),
        val idleSince: Long? = null
    ) {
        val isConnected = height != 0
    }

    private var state = State(0)

    private var runJob: Job? = null
    private var timerJob: Job? = null

    init {
        logger.info { "initializing electrum watcher" }

        suspend fun processScripHashHistory(history: List) = runCatching {
            val txs = history.filter { it.blockHeight >= -1 }.mapNotNull { client.getTx(it.txid) }

            // WatchSpent
            txs.forEach { tx ->
                val outpoints = tx.txIn.map { it.outPoint }
                outpoints.forEach { outPoint ->
                    state.watches
                        .filterIsInstance()
                        .filter { it.txId == outPoint.txid && it.outputIndex == outPoint.index.toInt() }
                        .map { w ->
                            logger.info { "output ${w.txId}:${w.outputIndex} spent by transaction ${tx.txid}" }
                            _notificationsFlow.emit(WatchEventSpent(w.channelId, w.event, tx))
                        }
                }
            }

            // WatchConfirmed
            val txMap = txs.associateBy { it.txid }
            history.filter { it.blockHeight > 0 }.forEach { item ->
                val triggered = state.watches
                    .filterIsInstance()
                    .filter { it.txId == item.txid }
                    .filter { state.height - item.blockHeight + 1 >= it.minDepth }
                triggered.forEach { w ->
                    client.getMerkle(w.txId, item.blockHeight)?.let { merkle ->
                        val confirmations = state.height - merkle.block_height + 1
                        logger.info { "txid=${w.txId} had confirmations=$confirmations in block=${merkle.block_height} pos=${merkle.pos}" }
                        _notificationsFlow.emit(WatchEventConfirmed(w.channelId, w.event, merkle.block_height, merkle.pos, txMap[w.txId]!!))

                        // check whether we have transactions to publish
                        when (val event = w.event) {
                            is BITCOIN_PARENT_TX_CONFIRMED -> {
                                val tx = event.childTx
                                logger.info { "parent tx of txid=${tx.txid} has been confirmed" }
                                val cltvTimeout = Scripts.cltvTimeout(tx)
                                val csvTimeout = Scripts.csvTimeout(tx)
                                val absTimeout = max(merkle.block_height + csvTimeout, cltvTimeout)
                                state = if (absTimeout > state.height) {
                                    logger.info { "delaying publication of txid=${tx.txid} until block=$absTimeout (curblock=${state.height})" }
                                    val block2tx = state.block2tx + (absTimeout to state.block2tx.getOrElse(absTimeout) { setOf() } + tx)
                                    state.copy(block2tx = block2tx)
                                } else {
                                    client.broadcastTransaction(tx)
                                    state.copy(sent = state.sent + tx)
                                }
                            }
                            else -> {}
                        }
                    }
                }
                state = state.copy(watches = state.watches - triggered.toSet())
            }
        }

        suspend fun processScripHashSubscriptionResponse(response: ScriptHashSubscriptionResponse) = runCatching {
            val existingStatus = state.scriptHashStatus[response.scriptHash]
            if (response.status.isNotEmpty() && response.status != existingStatus) {
                state = state.copy(scriptHashStatus = state.scriptHashStatus + (response.scriptHash to response.status))
                val history = client.getScriptHashHistory(response.scriptHash)
                processScripHashHistory(history)
                state = state.copy(idleSince = currentTimestampMillis())
            }
        }

        suspend fun addWatch(watch: Watch) {
            val scriptHash = when (watch) {
                is WatchSpent -> {
                    val (_, txid, outputIndex, publicKeyScript, _) = watch
                    val scriptHash = ElectrumClient.computeScriptHash(publicKeyScript)
                    logger.info { "added watch-spent on output=$txid:$outputIndex scriptHash=$scriptHash event=${watch.event}" }
                    scriptHash
                }

                is WatchConfirmed -> {
                    val (_, txid, publicKeyScript, _) = watch
                    val scriptHash = ElectrumClient.computeScriptHash(publicKeyScript)
                    logger.info { "added watch-confirmed on txid=$txid scriptHash=$scriptHash event=${watch.event}" }
                    scriptHash
                }
            }
            state = state.copy(
                watches = state.watches + watch, scriptHashSubscriptions = state.scriptHashSubscriptions + scriptHash
            )
            if (state.isConnected) {
                val response = client.startScriptHashSubscription(scriptHash)
                processScripHashSubscriptionResponse(response)
            }
        }

        fun startTimer() {
            if (timerJob != null) return

            val timeMillis: Long = 2L * 1_000 // fire timer every 2 seconds
            timerJob = launch {
                delay(timeMillis)
                while (isActive) {
                    mailbox.send(WatcherCommand.NotifyIfReady)
                    delay(timeMillis)
                }
            }
        }

        fun stopTimer() {
            timerJob?.cancel()
            timerJob = null
        }

        runJob = launch {
            mailbox.consumeAsFlow().collect { cmd ->
                when (cmd) {
                    is WatcherCommand.ProcessConnectionStatus -> {
                        when (cmd.status) {
                            is ElectrumConnectionStatus.Connecting -> {}

                            is ElectrumConnectionStatus.Connected -> {
                                state = state.copy(height = cmd.status.height)
                                // reset all subscriptions
                                state = state.copy(scriptHashSubscriptions = setOf(), scriptHashStatus = mapOf())
                                state.watches.forEach { addWatch(it) }

                                // handle pending publish commands
                                state.publishQueue.forEach { publish(it) }
                                state = state.copy(publishQueue = setOf())
                                startTimer()
                            }

                            is ElectrumConnectionStatus.Closed -> {
                                state = state.copy(height = 0, scriptHashSubscriptions = setOf(), scriptHashStatus = mapOf(), idleSince = null)
                                stopTimer()
                            }
                        }
                    }

                    is WatcherCommand.ProcessNotification -> {
                        when (cmd.notification) {
                            is ScriptHashSubscriptionResponse -> {
                                processScripHashSubscriptionResponse(cmd.notification)
                            }

                            is HeaderSubscriptionResponse -> {
                                logger.info { "got new tip ${cmd.notification}" }
                                state = state.copy(height = cmd.notification.blockHeight)

                                state.watches.filterIsInstance().forEach { watch ->
                                    val scriptHash = ElectrumClient.computeScriptHash(watch.publicKeyScript)
                                    val history = client.getScriptHashHistory(scriptHash)
                                    processScripHashHistory(history)
                                }

                                val toPublish = state.block2tx.filterKeys { it <= cmd.notification.blockHeight }
                                val txs = toPublish.values.flatten()
                                txs.forEach {
                                    logger.info { "publishing tx ${it.txid}" }
                                    client.broadcastTransaction(it)
                                }
                                state = state.copy(block2tx = state.block2tx - toPublish.keys, sent = state.sent + txs, idleSince = currentTimestampMillis())
                                logger.debug { "Watcher has processed new tip" }
                            }

                            else -> {}
                        }
                    }

                    is WatcherCommand.AddWatch -> addWatch(cmd.watch)

                    is WatcherCommand.Publish -> {
                        if (!state.isConnected) {
                            state = state.copy(publishQueue = state.publishQueue + cmd.tx)
                        } else {
                            val tx = cmd.tx
                            val blockCount = state.height
                            val cltvTimeout = Scripts.cltvTimeout(tx)
                            val csvTimeout = Scripts.csvTimeout(tx)
                            when {
                                csvTimeout > 0 -> {
                                    require(tx.txIn.size == 1) { "watcher only supports tx with 1 input, this tx has ${tx.txIn.size} inputs" }
                                    val parentTxid = tx.txIn[0].outPoint.txid
                                    logger.info { "txid=${tx.txid} has a relative timeout of $csvTimeout blocks, watching parenttxid=$parentTxid tx=$tx" }
                                    val parentPublicKeyScript = WatchConfirmed.extractPublicKeyScript(tx.txIn.first().witness)
                                    addWatch(WatchConfirmed(ByteVector32.Zeroes, parentTxid, parentPublicKeyScript, csvTimeout, BITCOIN_PARENT_TX_CONFIRMED(tx)))
                                }

                                cltvTimeout > blockCount -> {
                                    logger.info { "delaying publication of txid=${tx.txid} until block=$cltvTimeout (curblock=$blockCount)" }
                                    val block2tx = state.block2tx + (cltvTimeout to state.block2tx.getOrElse(cltvTimeout) { setOf() } + tx)
                                    state = state.copy(block2tx = block2tx)
                                }

                                else -> {
                                    logger.info { "publishing tx=[${tx.txid} / $tx]" }
                                    client.broadcastTransaction(tx)
                                    state = state.copy(sent = state.sent + tx)
                                }
                            }
                        }
                    }

                    is WatcherCommand.NotifyIfReady -> {
                        if (state.isConnected) {
                            state.idleSince?.let {
                                val now = currentTimestampMillis()
                                if (now > it + 5000) {
                                    // no requests in progress and watcher has been idle for more than 5s
                                    _uptodateFlow.emit(now)
                                }
                            }
                        }
                    }
                }
            }
        }

        launch {
            client.notifications.collect {
                mailbox.send(WatcherCommand.ProcessNotification(it))
            }
        }

        launch {
            client.connectionStatus.collect {
                mailbox.send(WatcherCommand.ProcessConnectionStatus(it))
            }
        }
    }

    fun stop() {
        logger.info { "electrum watcher stopping" }
        // Cancel event consumer
        runJob?.cancel()
        // Cancel up-to-date timer
        // Cancel event channels
        mailbox.cancel()
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy