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

commonMain.fr.acinq.lightning.channel.states.Negotiating.kt Maven / Gradle / Ivy

There is a newer version: 1.8.4
Show newest version
package fr.acinq.lightning.channel.states

import fr.acinq.bitcoin.Transaction
import fr.acinq.bitcoin.updated
import fr.acinq.lightning.blockchain.*
import fr.acinq.lightning.channel.*
import fr.acinq.lightning.channel.states.Channel.MAX_NEGOTIATION_ITERATIONS
import fr.acinq.lightning.transactions.Transactions.TransactionWithInputInfo.ClosingTx
import fr.acinq.lightning.utils.Either
import fr.acinq.lightning.utils.msat
import fr.acinq.lightning.wire.ClosingSigned
import fr.acinq.lightning.wire.ClosingSignedTlv
import fr.acinq.lightning.wire.Error
import fr.acinq.lightning.wire.Shutdown

data class Negotiating(
    override val commitments: Commitments,
    val localShutdown: Shutdown,
    val remoteShutdown: Shutdown,
    val closingTxProposed: List>, // one list for every negotiation (there can be several in case of disconnection)
    val bestUnpublishedClosingTx: ClosingTx?,
    val closingFeerates: ClosingFeerates?
) : ChannelStateWithCommitments() {
    init {
        require(closingTxProposed.isNotEmpty()) { "there must always be a list for the current negotiation" }
        require(!commitments.params.localParams.isInitiator || !closingTxProposed.any { it.isEmpty() }) { "initiator must have at least one closing signature for every negotiation attempt because it initiates the closing" }
    }

    override fun updateCommitments(input: Commitments): ChannelStateWithCommitments = this.copy(commitments = input)

    override fun ChannelContext.processInternal(cmd: ChannelCommand): Pair> {
        return when (cmd) {
            is ChannelCommand.MessageReceived -> when (cmd.message) {
                is ClosingSigned -> {
                    val remoteClosingFee = cmd.message.feeSatoshis
                    logger.info { "received closing fee=$remoteClosingFee" }
                    when (val result =
                        Helpers.Closing.checkClosingSignature(channelKeys(), commitments.latest, localShutdown.scriptPubKey.toByteArray(), remoteShutdown.scriptPubKey.toByteArray(), cmd.message.feeSatoshis, cmd.message.signature)) {
                        is Either.Left -> handleLocalError(cmd, result.value)
                        is Either.Right -> {
                            val (signedClosingTx, closingSignedRemoteFees) = result.value
                            val lastLocalClosingSigned = closingTxProposed.last().lastOrNull()?.localClosingSigned
                            when {
                                lastLocalClosingSigned?.feeSatoshis == remoteClosingFee -> {
                                    logger.info { "they accepted our fee, publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                    completeMutualClose(signedClosingTx, null)
                                }
                                closingTxProposed.flatten().size >= MAX_NEGOTIATION_ITERATIONS -> {
                                    logger.warning { "could not agree on closing fees after $MAX_NEGOTIATION_ITERATIONS iterations, publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                    completeMutualClose(signedClosingTx, closingSignedRemoteFees)
                                }
                                lastLocalClosingSigned?.tlvStream?.get()?.let { it.min <= remoteClosingFee && remoteClosingFee <= it.max } == true -> {
                                    val localFeeRange = lastLocalClosingSigned.tlvStream.get()!!
                                    logger.info { "they chose closing fee=$remoteClosingFee within our fee range (min=${localFeeRange.max} max=${localFeeRange.max}), publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                    completeMutualClose(signedClosingTx, closingSignedRemoteFees)
                                }
                                commitments.latest.localCommit.spec.toLocal == 0.msat -> {
                                    logger.info { "we have nothing at stake, publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                    completeMutualClose(signedClosingTx, closingSignedRemoteFees)
                                }
                                else -> {
                                    val theirFeeRange = cmd.message.tlvStream.get()
                                    val ourFeeRange = closingFeerates ?: ClosingFeerates(currentOnChainFeerates.mutualCloseFeerate)
                                    when {
                                        theirFeeRange != null && !commitments.params.localParams.isInitiator -> {
                                            // if we are not the initiator and they proposed a fee range, we pick a value in that range and they should accept it without further negotiation
                                            // we don't care much about the closing fee since they're paying it (not us) and we can use CPFP if we want to speed up confirmation
                                            val closingFees = Helpers.Closing.firstClosingFee(commitments.latest, localShutdown.scriptPubKey, remoteShutdown.scriptPubKey, ourFeeRange)
                                            val closingFee = when {
                                                closingFees.preferred > theirFeeRange.max -> theirFeeRange.max
                                                // if we underestimate the fee, then we're happy with whatever they propose (it will confirm more quickly and we're not paying it)
                                                closingFees.preferred < remoteClosingFee -> remoteClosingFee
                                                else -> closingFees.preferred
                                            }
                                            if (closingFee == remoteClosingFee) {
                                                logger.info { "accepting their closing fee=$remoteClosingFee, publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                                completeMutualClose(signedClosingTx, closingSignedRemoteFees)
                                            } else {
                                                val (closingTx, closingSigned) = Helpers.Closing.makeClosingTx(
                                                    channelKeys(),
                                                    commitments.latest,
                                                    localShutdown.scriptPubKey.toByteArray(),
                                                    remoteShutdown.scriptPubKey.toByteArray(),
                                                    ClosingFees(closingFee, theirFeeRange.min, theirFeeRange.max)
                                                )
                                                logger.info { "proposing closing fee=${closingSigned.feeSatoshis}" }
                                                val closingProposed1 = closingTxProposed.updated(
                                                    closingTxProposed.lastIndex,
                                                    closingTxProposed.last() + listOf(ClosingTxProposed(closingTx, closingSigned))
                                                )
                                                val nextState = [email protected](
                                                    commitments = commitments.copy(remoteChannelData = cmd.message.channelData),
                                                    closingTxProposed = closingProposed1,
                                                    bestUnpublishedClosingTx = signedClosingTx
                                                )
                                                val actions = listOf(ChannelAction.Storage.StoreState(nextState), ChannelAction.Message.Send(closingSigned))
                                                Pair(nextState, actions)
                                            }
                                        }
                                        else -> {
                                            val (closingTx, closingSigned) = run {
                                                // if we are not the initiator and we were waiting for them to send their first closing_signed, we compute our firstClosingFee, otherwise we use the last one we sent
                                                val localClosingFees = Helpers.Closing.firstClosingFee(commitments.latest, localShutdown.scriptPubKey, remoteShutdown.scriptPubKey, ourFeeRange)
                                                val nextPreferredFee = Helpers.Closing.nextClosingFee(lastLocalClosingSigned?.feeSatoshis ?: localClosingFees.preferred, remoteClosingFee)
                                                Helpers.Closing.makeClosingTx(
                                                    channelKeys(),
                                                    commitments.latest,
                                                    localShutdown.scriptPubKey.toByteArray(),
                                                    remoteShutdown.scriptPubKey.toByteArray(),
                                                    localClosingFees.copy(preferred = nextPreferredFee)
                                                )
                                            }
                                            when {
                                                lastLocalClosingSigned?.feeSatoshis == closingSigned.feeSatoshis -> {
                                                    // next computed fee is the same than the one we previously sent (probably because of rounding)
                                                    logger.info { "accepting their closing fee=$remoteClosingFee, publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                                    completeMutualClose(signedClosingTx, null)
                                                }
                                                closingSigned.feeSatoshis == remoteClosingFee -> {
                                                    logger.info { "we have converged, publishing closing tx: closingTxId=${signedClosingTx.tx.txid}" }
                                                    completeMutualClose(signedClosingTx, closingSigned)
                                                }
                                                else -> {
                                                    logger.info { "proposing closing fee=${closingSigned.feeSatoshis}" }
                                                    val closingProposed1 = closingTxProposed.updated(
                                                        closingTxProposed.lastIndex,
                                                        closingTxProposed.last() + listOf(ClosingTxProposed(closingTx, closingSigned))
                                                    )
                                                    val nextState = [email protected](
                                                        commitments = commitments.copy(remoteChannelData = cmd.message.channelData),
                                                        closingTxProposed = closingProposed1,
                                                        bestUnpublishedClosingTx = signedClosingTx
                                                    )
                                                    val actions = listOf(ChannelAction.Storage.StoreState(nextState), ChannelAction.Message.Send(closingSigned))
                                                    Pair(nextState, actions)
                                                }
                                            }
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
                is Error -> handleRemoteError(cmd.message)
                else -> unhandled(cmd)
            }
            is ChannelCommand.WatchReceived -> when (val watch = cmd.watch) {
                is WatchEventConfirmed -> updateFundingTxStatus(watch)
                is WatchEventSpent -> when {
                    watch.event is BITCOIN_FUNDING_SPENT && closingTxProposed.flatten().any { it.unsignedTx.tx.txid == watch.tx.txid } -> {
                        // they can publish a closing tx with any sig we sent them, even if we are not done negotiating
                        logger.info { "closing tx published: closingTxId=${watch.tx.txid}" }
                        val closingTx = getMutualClosePublished(watch.tx)
                        val nextState = Closing(
                            commitments,
                            waitingSinceBlock = currentBlockHeight.toLong(),
                            mutualCloseProposed = closingTxProposed.flatten().map { it.unsignedTx },
                            mutualClosePublished = listOf(closingTx)
                        )
                        val actions = listOf(
                            ChannelAction.Storage.StoreState(nextState),
                            ChannelAction.Blockchain.PublishTx(closingTx),
                            ChannelAction.Blockchain.SendWatch(WatchConfirmed(channelId, watch.tx, staticParams.nodeParams.minDepthBlocks.toLong(), BITCOIN_TX_CONFIRMED(watch.tx)))
                        )
                        Pair(nextState, actions)
                    }
                    else -> handlePotentialForceClose(watch)
                }
            }
            is ChannelCommand.Commitment.CheckHtlcTimeout -> checkHtlcTimeout()
            is ChannelCommand.Commitment -> unhandled(cmd)
            is ChannelCommand.Htlc.Add -> handleCommandError(cmd, ChannelUnavailable(channelId))
            is ChannelCommand.Htlc -> unhandled(cmd)
            is ChannelCommand.Close.ForceClose -> handleLocalError(cmd, ForcedLocalCommit(channelId))
            is ChannelCommand.Close.MutualClose -> handleCommandError(cmd, ClosingAlreadyInProgress(channelId))
            is ChannelCommand.Init -> unhandled(cmd)
            is ChannelCommand.Funding -> unhandled(cmd)
            is ChannelCommand.Closing -> unhandled(cmd)
            is ChannelCommand.Connected -> unhandled(cmd)
            is ChannelCommand.Disconnected -> Pair(Offline(this@Negotiating), listOf())
        }
    }

    /** Return full information about a known closing tx. */
    internal fun getMutualClosePublished(tx: Transaction): ClosingTx {
        // they can publish a closing tx with any sig we sent them, even if we are not done negotiating
        // they added their signature, so we use their version of the transaction
        return closingTxProposed.flatten().first { it.unsignedTx.tx.txid == tx.txid }.unsignedTx.copy(tx = tx)
    }

    private fun ChannelContext.completeMutualClose(signedClosingTx: ClosingTx, closingSigned: ClosingSigned?): Pair> {
        val nextState = Closing(
            commitments,
            waitingSinceBlock = currentBlockHeight.toLong(),
            mutualCloseProposed = closingTxProposed.flatten().map { it.unsignedTx },
            mutualClosePublished = listOf(signedClosingTx)
        )
        val actions = buildList {
            add(ChannelAction.Storage.StoreState(nextState))
            closingSigned?.let { add(ChannelAction.Message.Send(it)) }
            add(ChannelAction.Blockchain.PublishTx(signedClosingTx))
            add(ChannelAction.Blockchain.SendWatch(WatchConfirmed(channelId, signedClosingTx.tx, staticParams.nodeParams.minDepthBlocks.toLong(), BITCOIN_TX_CONFIRMED(signedClosingTx.tx))))
        }
        return Pair(nextState, actions)
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy