commonMain.fr.acinq.lightning.channel.Commitments.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of lightning-kmp Show documentation
Show all versions of lightning-kmp Show documentation
A Kotlin Multiplatform implementation of the Lightning Network
package fr.acinq.lightning.channel
import fr.acinq.bitcoin.*
import fr.acinq.bitcoin.Crypto.sha256
import fr.acinq.lightning.CltvExpiryDelta
import fr.acinq.lightning.Feature
import fr.acinq.lightning.MilliSatoshi
import fr.acinq.lightning.blockchain.fee.FeeratePerKw
import fr.acinq.lightning.blockchain.fee.FeerateTolerance
import fr.acinq.lightning.channel.states.Channel
import fr.acinq.lightning.channel.states.ChannelContext
import fr.acinq.lightning.crypto.Bolt3Derivation.deriveForCommitment
import fr.acinq.lightning.crypto.Bolt3Derivation.deriveForRevocation
import fr.acinq.lightning.crypto.KeyManager
import fr.acinq.lightning.crypto.ShaChain
import fr.acinq.lightning.payment.OutgoingPaymentPacket
import fr.acinq.lightning.transactions.CommitmentSpec
import fr.acinq.lightning.transactions.Transactions
import fr.acinq.lightning.transactions.Transactions.TransactionWithInputInfo.CommitTx
import fr.acinq.lightning.transactions.Transactions.TransactionWithInputInfo.HtlcTx
import fr.acinq.lightning.transactions.Transactions.commitTxFee
import fr.acinq.lightning.transactions.Transactions.commitTxFeeMsat
import fr.acinq.lightning.transactions.Transactions.htlcOutputFee
import fr.acinq.lightning.transactions.Transactions.makeCommitTxOutputs
import fr.acinq.lightning.transactions.Transactions.offeredHtlcTrimThreshold
import fr.acinq.lightning.transactions.Transactions.receivedHtlcTrimThreshold
import fr.acinq.lightning.transactions.incomings
import fr.acinq.lightning.transactions.outgoings
import fr.acinq.lightning.utils.*
import fr.acinq.lightning.wire.*
import kotlin.math.min
/** Static channel parameters shared by all commitments. */
data class ChannelParams(
val channelId: ByteVector32,
val channelConfig: ChannelConfig,
val channelFeatures: ChannelFeatures,
val localParams: LocalParams, val remoteParams: RemoteParams,
val channelFlags: Byte
) {
init {
require(channelConfig.hasOption(ChannelConfigOption.FundingPubKeyBasedChannelKeyPath)) { "FundingPubKeyBasedChannelKeyPath option must be enabled" }
}
fun updateFeatures(localInit: Init, remoteInit: Init) = this.copy(
localParams = localParams.copy(features = localInit.features),
remoteParams = remoteParams.copy(features = remoteInit.features)
)
}
data class LocalChanges(val proposed: List, val signed: List, val acked: List) {
val all: List get() = proposed + signed + acked
}
data class RemoteChanges(val proposed: List, val acked: List, val signed: List) {
val all: List get() = proposed + signed + acked
}
/** Changes are applied to all commitments, and must be be valid for all commitments. */
data class CommitmentChanges(val localChanges: LocalChanges, val remoteChanges: RemoteChanges, val localNextHtlcId: Long, val remoteNextHtlcId: Long) {
fun addLocalProposal(proposal: UpdateMessage): CommitmentChanges = copy(localChanges = localChanges.copy(proposed = localChanges.proposed + proposal))
fun addRemoteProposal(proposal: UpdateMessage): CommitmentChanges = copy(remoteChanges = remoteChanges.copy(proposed = remoteChanges.proposed + proposal))
fun localHasUnsignedOutgoingHtlcs(): Boolean = localChanges.proposed.find { it is UpdateAddHtlc } != null
fun remoteHasUnsignedOutgoingHtlcs(): Boolean = remoteChanges.proposed.find { it is UpdateAddHtlc } != null
fun localHasUnsignedOutgoingUpdateFee(): Boolean = localChanges.proposed.find { it is UpdateFee } != null
fun remoteHasUnsignedOutgoingUpdateFee(): Boolean = remoteChanges.proposed.find { it is UpdateFee } != null
fun localHasChanges(): Boolean = remoteChanges.acked.isNotEmpty() || localChanges.proposed.isNotEmpty()
fun remoteHasChanges(): Boolean = localChanges.acked.isNotEmpty() || remoteChanges.proposed.isNotEmpty()
companion object {
fun init(): CommitmentChanges = CommitmentChanges(LocalChanges(listOf(), listOf(), listOf()), RemoteChanges(listOf(), listOf(), listOf()), 0, 0)
fun alreadyProposed(changes: List, id: Long): Boolean = changes.any {
when (it) {
is UpdateFulfillHtlc -> id == it.id
is UpdateFailHtlc -> id == it.id
is UpdateFailMalformedHtlc -> id == it.id
else -> false
}
}
}
}
data class HtlcTxAndSigs(val txinfo: HtlcTx, val localSig: ByteVector64, val remoteSig: ByteVector64)
data class PublishableTxs(val commitTx: CommitTx, val htlcTxsAndSigs: List)
/** The local commitment maps to a commitment transaction that we can sign and broadcast if necessary. */
data class LocalCommit(val index: Long, val spec: CommitmentSpec, val publishableTxs: PublishableTxs)
/** The remote commitment maps to a commitment transaction that only our peer can sign and broadcast. */
data class RemoteCommit(val index: Long, val spec: CommitmentSpec, val txid: ByteVector32, val remotePerCommitmentPoint: PublicKey) {
fun sign(channelKeys: KeyManager.ChannelKeys, params: ChannelParams, fundingTxIndex: Long, remoteFundingPubKey: PublicKey, commitInput: Transactions.InputInfo): CommitSig {
val (remoteCommitTx, htlcTxs) = Commitments.makeRemoteTxs(channelKeys, index, params.localParams, params.remoteParams, fundingTxIndex = fundingTxIndex, remoteFundingPubKey = remoteFundingPubKey, commitInput, remotePerCommitmentPoint = remotePerCommitmentPoint, spec)
val sig = Transactions.sign(remoteCommitTx, channelKeys.fundingKey(fundingTxIndex))
// we sign our peer's HTLC txs with SIGHASH_SINGLE || SIGHASH_ANYONECANPAY
val sortedHtlcsTxs = htlcTxs.sortedBy { it.input.outPoint.index }
val htlcSigs = sortedHtlcsTxs.map { Transactions.sign(it, channelKeys.htlcKey.deriveForCommitment(remotePerCommitmentPoint), SigHash.SIGHASH_SINGLE or SigHash.SIGHASH_ANYONECANPAY) }
return CommitSig(params.channelId, sig, htlcSigs.toList())
}
fun sign(channelKeys: KeyManager.ChannelKeys, params: ChannelParams, signingSession: InteractiveTxSigningSession): CommitSig =
sign(channelKeys, params, signingSession.fundingTxIndex, signingSession.fundingParams.remoteFundingPubkey, signingSession.commitInput)
}
/** We have the next remote commit when we've sent our commit_sig but haven't yet received their revoke_and_ack. */
data class NextRemoteCommit(val sig: CommitSig, val commit: RemoteCommit)
sealed class LocalFundingStatus {
abstract val signedTx: Transaction?
abstract val txId: ByteVector32
abstract val fee: Satoshi
data class UnconfirmedFundingTx(val sharedTx: SignedSharedTransaction, val fundingParams: InteractiveTxParams, val createdAt: Long) : LocalFundingStatus() {
override val signedTx: Transaction? = sharedTx.signedTx
override val txId: ByteVector32 = sharedTx.localSigs.txId
override val fee: Satoshi = sharedTx.tx.fees
}
data class ConfirmedFundingTx(override val signedTx: Transaction, override val fee: Satoshi, val localSigs: TxSignatures) : LocalFundingStatus() {
override val txId: ByteVector32 = signedTx.txid
}
}
sealed class RemoteFundingStatus {
object NotLocked : RemoteFundingStatus()
object Locked : RemoteFundingStatus()
}
/** A minimal commitment for a given funding tx. */
data class Commitment(
val fundingTxIndex: Long,
val remoteFundingPubkey: PublicKey,
val localFundingStatus: LocalFundingStatus, val remoteFundingStatus: RemoteFundingStatus,
val localCommit: LocalCommit, val remoteCommit: RemoteCommit, val nextRemoteCommit: NextRemoteCommit?
) {
val commitInput = localCommit.publishableTxs.commitTx.input
val fundingTxId: ByteVector32 = commitInput.outPoint.txid
val fundingAmount: Satoshi = commitInput.txOut.amount
fun localChannelReserve(params: ChannelParams): Satoshi = when {
params.channelFeatures.hasFeature(Feature.ZeroReserveChannels) && !params.localParams.isInitiator -> 0.sat
else -> (fundingAmount / 100).max(params.remoteParams.dustLimit)
}
fun remoteChannelReserve(params: ChannelParams): Satoshi = when {
params.channelFeatures.hasFeature(Feature.ZeroReserveChannels) && params.localParams.isInitiator -> 0.sat
else -> (fundingAmount / 100).max(params.localParams.dustLimit)
}
// NB: when computing availableBalanceForSend and availableBalanceForReceive, the initiator keeps an extra buffer on top
// of its usual channel reserve to avoid getting channels stuck in case the on-chain feerate increases (see
// https://github.com/lightningnetwork/lightning-rfc/issues/728 for details).
//
// This extra buffer (which we call "initiator fee buffer") is calculated as follows:
// 1) Simulate a x2 feerate increase and compute the corresponding commit tx fee (note that it may trim some HTLCs)
// 2) Add the cost of adding a new untrimmed HTLC at that increased feerate. This ensures that we'll be able to
// actually use the channel to add new HTLCs if the feerate doubles.
//
// If for example the current feerate is 1000 sat/kw, the dust limit 546 sat, and we have 3 pending outgoing HTLCs for
// respectively 1250 sat, 2000 sat and 2500 sat.
// commit tx fee = commitWeight * feerate + 3 * htlcOutputWeight * feerate = 724 * 1000 + 3 * 172 * 1000 = 1240 sat
// To calculate the initiator fee buffer, we first double the feerate and calculate the corresponding commit tx fee.
// By doubling the feerate, the first HTLC becomes trimmed so the result is: 724 * 2000 + 2 * 172 * 2000 = 2136 sat
// We then add the additional fee for a potential new untrimmed HTLC: 172 * 2000 = 344 sat
// The initiator fee buffer is 2136 + 344 = 2480 sat
//
// If there are many pending HTLCs that are only slightly above the trim threshold, the initiator fee buffer may be
// smaller than the current commit tx fee because those HTLCs will be trimmed and the commit tx weight will decrease.
// For example if we have 10 outgoing HTLCs of 1250 sat:
// - commit tx fee = 724 * 1000 + 10 * 172 * 1000 = 2444 sat
// - commit tx fee at twice the feerate = 724 * 2000 = 1448 sat (all HTLCs have been trimmed)
// - cost of an additional untrimmed HTLC = 172 * 2000 = 344 sat
// - initiator fee buffer = 1448 + 344 = 1792 sat
// In that case the current commit tx fee is higher than the initiator fee buffer and will dominate the balance restrictions.
fun availableBalanceForSend(params: ChannelParams, changes: CommitmentChanges): MilliSatoshi {
// we need to base the next current commitment on the last sig we sent, even if we didn't yet receive their revocation
val remoteCommit1 = nextRemoteCommit?.commit ?: remoteCommit
val reduced = CommitmentSpec.reduce(remoteCommit1.spec, changes.remoteChanges.acked, changes.localChanges.proposed)
val balanceNoFees = (reduced.toRemote - localChannelReserve(params).toMilliSatoshi()).coerceAtLeast(0.msat)
return if (params.localParams.isInitiator) {
// The initiator always pays the on-chain fees, so we must subtract that from the amount we can send.
val commitFees = commitTxFeeMsat(params.remoteParams.dustLimit, reduced)
// the initiator needs to keep a "initiator fee buffer" (see explanation above)
val initiatorFeeBuffer = commitTxFeeMsat(params.remoteParams.dustLimit, reduced.copy(feerate = reduced.feerate * 2)) + htlcOutputFee(reduced.feerate * 2)
val amountToReserve = commitFees.coerceAtLeast(initiatorFeeBuffer)
if (balanceNoFees - amountToReserve < offeredHtlcTrimThreshold(params.remoteParams.dustLimit, reduced).toMilliSatoshi()) {
// htlc will be trimmed
(balanceNoFees - amountToReserve).coerceAtLeast(0.msat)
} else {
// htlc will have an output in the commitment tx, so there will be additional fees.
val commitFees1 = commitFees + htlcOutputFee(reduced.feerate)
// we take the additional fees for that htlc output into account in the fee buffer at a x2 feerate increase
val initiatorFeeBuffer1 = initiatorFeeBuffer + htlcOutputFee(reduced.feerate * 2)
val amountToReserve1 = commitFees1.coerceAtLeast(initiatorFeeBuffer1)
(balanceNoFees - amountToReserve1).coerceAtLeast(0.msat)
}
} else {
// The non-initiator doesn't pay on-chain fees.
balanceNoFees
}
}
fun availableBalanceForReceive(params: ChannelParams, changes: CommitmentChanges): MilliSatoshi {
val reduced = CommitmentSpec.reduce(localCommit.spec, changes.localChanges.acked, changes.remoteChanges.proposed)
val balanceNoFees = (reduced.toRemote - remoteChannelReserve(params).toMilliSatoshi()).coerceAtLeast(0.msat)
return if (params.localParams.isInitiator) {
// The non-initiator doesn't pay on-chain fees so we don't take those into account when receiving.
balanceNoFees
} else {
// The initiator always pays the on-chain fees, so we must subtract that from the amount we can receive.
val commitFees = commitTxFeeMsat(params.localParams.dustLimit, reduced)
// we expected the initiator to keep a "initiator fee buffer" (see explanation above)
val initiatorFeeBuffer = commitTxFeeMsat(params.localParams.dustLimit, reduced.copy(feerate = reduced.feerate * 2)) + htlcOutputFee(reduced.feerate * 2)
val amountToReserve = commitFees.coerceAtLeast(initiatorFeeBuffer)
if (balanceNoFees - amountToReserve < receivedHtlcTrimThreshold(params.localParams.dustLimit, reduced).toMilliSatoshi()) {
// htlc will be trimmed
(balanceNoFees - amountToReserve).coerceAtLeast(0.msat)
} else {
// htlc will have an output in the commitment tx, so there will be additional fees.
val commitFees1 = commitFees + htlcOutputFee(reduced.feerate)
// we take the additional fees for that htlc output into account in the fee buffer at a x2 feerate increase
val initiatorFeeBuffer1 = initiatorFeeBuffer + htlcOutputFee(reduced.feerate * 2)
val amountToReserve1 = commitFees1.coerceAtLeast(initiatorFeeBuffer1)
(balanceNoFees - amountToReserve1).coerceAtLeast(0.msat)
}
}
}
fun hasNoPendingHtlcs(): Boolean = localCommit.spec.htlcs.isEmpty() && remoteCommit.spec.htlcs.isEmpty() && nextRemoteCommit == null
fun hasNoPendingHtlcsOrFeeUpdate(changes: CommitmentChanges): Boolean {
val hasNoPendingFeeUpdate = (changes.localChanges.signed + changes.localChanges.acked + changes.remoteChanges.signed + changes.remoteChanges.acked).find { it is UpdateFee } == null
return hasNoPendingHtlcs() && hasNoPendingFeeUpdate
}
fun isIdle(changes: CommitmentChanges): Boolean = hasNoPendingHtlcs() && changes.localChanges.all.isEmpty() && changes.remoteChanges.all.isEmpty()
fun timedOutOutgoingHtlcs(blockHeight: Long): Set {
fun expired(add: UpdateAddHtlc) = blockHeight >= add.cltvExpiry.toLong()
val thisCommitAdds = localCommit.spec.htlcs.outgoings().filter(::expired).toSet() + remoteCommit.spec.htlcs.incomings().filter(::expired).toSet()
return when (nextRemoteCommit) {
null -> thisCommitAdds
else -> thisCommitAdds + nextRemoteCommit.commit.spec.htlcs.incomings().filter(::expired).toSet()
}
}
/**
* Incoming HTLCs that are close to timing out are potentially dangerous. If we released the pre-image for those
* HTLCs, we need to get a remote signed updated commitment that removes this HTLC.
* Otherwise when we get close to the timeout, we risk an on-chain race condition between their HTLC timeout
* and our HTLC success in case of a force-close.
*/
fun almostTimedOutIncomingHtlcs(blockHeight: Long, fulfillSafety: CltvExpiryDelta, changes: CommitmentChanges): Set {
val relayedFulfills = changes.localChanges.all.filterIsInstance().map { it.id }.toSet()
return localCommit.spec.htlcs.incomings().filter { relayedFulfills.contains(it.id) && blockHeight >= (it.cltvExpiry - fulfillSafety).toLong() }.toSet()
}
fun getOutgoingHtlcCrossSigned(htlcId: Long): UpdateAddHtlc? {
val localSigned = (nextRemoteCommit?.commit ?: remoteCommit).spec.findIncomingHtlcById(htlcId) ?: return null
val remoteSigned = localCommit.spec.findOutgoingHtlcById(htlcId) ?: return null
require(localSigned.add == remoteSigned.add)
return localSigned.add
}
fun getIncomingHtlcCrossSigned(htlcId: Long): UpdateAddHtlc? {
val localSigned = (nextRemoteCommit?.commit ?: remoteCommit).spec.findOutgoingHtlcById(htlcId) ?: return null
val remoteSigned = localCommit.spec.findIncomingHtlcById(htlcId) ?: return null
require(localSigned.add == remoteSigned.add)
return localSigned.add
}
fun canSendAdd(amount: MilliSatoshi, params: ChannelParams, changes: CommitmentChanges): Either {
// we need to base the next current commitment on the last sig we sent, even if we didn't yet receive their revocation
val remoteCommit1 = nextRemoteCommit?.commit ?: remoteCommit
val reduced = CommitmentSpec.reduce(remoteCommit1.spec, changes.remoteChanges.acked, changes.localChanges.proposed)
// the HTLC we are about to create is outgoing, but from their point of view it is incoming
val outgoingHtlcs = reduced.htlcs.incomings()
// note that the initiator pays the fee, so if sender != initiator, both sides will have to afford this payment
val fees = commitTxFee(params.remoteParams.dustLimit, reduced)
// the initiator needs to keep an extra buffer to be able to handle a x2 feerate increase and an additional htlc to avoid
// getting the channel stuck (see https://github.com/lightningnetwork/lightning-rfc/issues/728).
val initiatorFeeBuffer = commitTxFeeMsat(params.remoteParams.dustLimit, reduced.copy(feerate = reduced.feerate * 2)) + htlcOutputFee(reduced.feerate * 2)
// NB: increasing the feerate can actually remove htlcs from the commit tx (if they fall below the trim threshold)
// which may result in a lower commit tx fee; this is why we take the max of the two.
val missingForSender = reduced.toRemote - localChannelReserve(params).toMilliSatoshi() - (if (params.localParams.isInitiator) fees.toMilliSatoshi().coerceAtLeast(initiatorFeeBuffer) else 0.msat)
// According to BOLT 2, we should also subtract the channel reserve from the calculation below.
// But this creates issues with splicing in the following scenario:
// - Alice opened a channel to Bob, and her balance is slightly above the reserve
// - Bob splices some funds in, which increases the size of the reserve since it is set to 1% by default
// - Alice is now below her reserve, so Bob is unable to send her any HTLC
// - The liquidity is mostly on Bob's side, but since he's unable to send HTLCs the channel is stuck
// We instead only check that the channel initiator is able to pay the fees for the commit tx.
// We are sending an outgoing HTLC, so once it's fulfilled it will increase their balance which is good for the channel reserve.
val missingForReceiver = reduced.toLocal - (if (params.localParams.isInitiator) 0.msat else fees.toMilliSatoshi())
if (missingForSender < 0.msat) {
val actualFees = if (params.localParams.isInitiator) fees else 0.sat
return Either.Left(InsufficientFunds(params.channelId, amount, -missingForSender.truncateToSatoshi(), localChannelReserve(params), actualFees))
} else if (missingForReceiver < 0.msat) {
if (params.localParams.isInitiator) {
// receiver is not the initiator; it is ok if it can't maintain its channel_reserve for now, as long as its balance is increasing, which is the case if it is receiving a payment
} else {
return Either.Left(RemoteCannotAffordFeesForNewHtlc(params.channelId, amount = amount, missing = -missingForReceiver.truncateToSatoshi(), fees = fees))
}
}
// README: we check against our peer's max_htlc_value_in_flight_msat parameter, as per the BOLTS, but also against our own setting
val htlcValueInFlight = outgoingHtlcs.map { it.amountMsat }.sum()
val maxHtlcValueInFlightMsat = min(params.remoteParams.maxHtlcValueInFlightMsat, params.localParams.maxHtlcValueInFlightMsat)
if (htlcValueInFlight.toLong() > maxHtlcValueInFlightMsat) {
return Either.Left(HtlcValueTooHighInFlight(params.channelId, maximum = maxHtlcValueInFlightMsat.toULong(), actual = htlcValueInFlight))
}
if (outgoingHtlcs.size > params.remoteParams.maxAcceptedHtlcs) {
return Either.Left(TooManyAcceptedHtlcs(params.channelId, maximum = params.remoteParams.maxAcceptedHtlcs.toLong()))
}
// README: this is not part of the LN Bolts: we also check against our own limit, to avoid creating commit txs that have too many outputs
if (outgoingHtlcs.size > params.localParams.maxAcceptedHtlcs) {
return Either.Left(TooManyOfferedHtlcs(params.channelId, maximum = params.localParams.maxAcceptedHtlcs.toLong()))
}
return Either.Right(Unit)
}
fun canReceiveAdd(amount: MilliSatoshi, params: ChannelParams, changes: CommitmentChanges): Either {
// let's compute the current commitment *as seen by us* including this change
val reduced = CommitmentSpec.reduce(localCommit.spec, changes.localChanges.acked, changes.remoteChanges.proposed)
val incomingHtlcs = reduced.htlcs.incomings()
// note that the initiator pays the fee, so if sender != initiator, both sides will have to afford this payment
val fees = commitTxFee(params.remoteParams.dustLimit, reduced)
// NB: we don't enforce the initiatorFeeReserve (see sendAdd) because it would confuse a remote initiator that doesn't have this mitigation in place
// We could enforce it once we're confident a large portion of the network implements it.
val missingForSender = reduced.toRemote - remoteChannelReserve(params).toMilliSatoshi() - (if (params.localParams.isInitiator) 0.sat else fees).toMilliSatoshi()
// We diverge from Bolt 2 and don't subtract the channel reserve: see `canSendAdd` for details.
val missingForReceiver = reduced.toLocal - (if (params.localParams.isInitiator) fees else 0.sat).toMilliSatoshi()
if (missingForSender < 0.sat) {
val actualFees = if (params.localParams.isInitiator) 0.sat else fees
return Either.Left(InsufficientFunds(params.channelId, amount, -missingForSender.truncateToSatoshi(), remoteChannelReserve(params), actualFees))
} else if (missingForReceiver < 0.sat) {
if (params.localParams.isInitiator) {
return Either.Left(CannotAffordFees(params.channelId, missing = -missingForReceiver.truncateToSatoshi(), reserve = localChannelReserve(params), fees = fees))
} else {
// receiver is not the initiator; it is ok if it can't maintain its channel_reserve for now, as long as its balance is increasing, which is the case if it is receiving a payment
}
}
val htlcValueInFlight = incomingHtlcs.map { it.amountMsat }.sum()
if (params.localParams.maxHtlcValueInFlightMsat < htlcValueInFlight.toLong()) {
return Either.Left(HtlcValueTooHighInFlight(params.channelId, maximum = params.localParams.maxHtlcValueInFlightMsat.toULong(), actual = htlcValueInFlight))
}
if (incomingHtlcs.size > params.localParams.maxAcceptedHtlcs) {
return Either.Left(TooManyAcceptedHtlcs(params.channelId, maximum = params.localParams.maxAcceptedHtlcs.toLong()))
}
return Either.Right(Unit)
}
fun canSendFee(params: ChannelParams, changes: CommitmentChanges): Either {
val reduced = CommitmentSpec.reduce(remoteCommit.spec, changes.remoteChanges.acked, changes.localChanges.proposed)
// a node cannot spend pending incoming htlcs, and need to keep funds above the reserve required by the counterparty, after paying the fee
// we look from remote's point of view, so if local is initiator remote doesn't pay the fees
val fees = commitTxFee(params.remoteParams.dustLimit, reduced)
val missing = reduced.toRemote.truncateToSatoshi() - localChannelReserve(params) - fees
return if (missing < 0.sat) {
Either.Left(CannotAffordFees(params.channelId, -missing, localChannelReserve(params), fees))
} else {
Either.Right(Unit)
}
}
fun canReceiveFee(params: ChannelParams, changes: CommitmentChanges): Either {
// let's compute the current commitment *as seen by us* including this change
// update_fee replace each other, so we can remove previous ones
val reduced = CommitmentSpec.reduce(localCommit.spec, changes.localChanges.acked, changes.remoteChanges.proposed)
// NB: we check that the initiator can afford this new fee even if spec allows to do it at next signature
// It is easier to do it here because under certain (race) conditions spec allows a lower-than-normal fee to be paid,
// and it would be tricky to check if the conditions are met at signing
// (it also means that we need to check the fee of the initial commitment tx somewhere)
val fees = commitTxFee(params.remoteParams.dustLimit, reduced)
val missing = reduced.toRemote.truncateToSatoshi() - remoteChannelReserve(params) - fees
return if (missing < 0.sat) {
Either.Left(CannotAffordFees(params.channelId, -missing, remoteChannelReserve(params), fees))
} else {
Either.Right(Unit)
}
}
fun sendCommit(channelKeys: KeyManager.ChannelKeys, params: ChannelParams, changes: CommitmentChanges, remoteNextPerCommitmentPoint: PublicKey, log: MDCLogger): Pair {
// remote commitment will include all local changes + remote acked changes
val spec = CommitmentSpec.reduce(remoteCommit.spec, changes.remoteChanges.acked, changes.localChanges.proposed)
val (remoteCommitTx, htlcTxs) = Commitments.makeRemoteTxs(channelKeys, commitTxNumber = remoteCommit.index + 1, params.localParams, params.remoteParams, fundingTxIndex = fundingTxIndex, remoteFundingPubKey = remoteFundingPubkey, commitInput, remotePerCommitmentPoint = remoteNextPerCommitmentPoint, spec)
val sig = Transactions.sign(remoteCommitTx, channelKeys.fundingKey(fundingTxIndex))
val sortedHtlcTxs: List = htlcTxs.sortedBy { it.input.outPoint.index }
// we sign our peer's HTLC txs with SIGHASH_SINGLE || SIGHASH_ANYONECANPAY
val htlcSigs = sortedHtlcTxs.map { Transactions.sign(it, channelKeys.htlcKey.deriveForCommitment(remoteNextPerCommitmentPoint), SigHash.SIGHASH_SINGLE or SigHash.SIGHASH_ANYONECANPAY) }
// NB: IN/OUT htlcs are inverted because this is the remote commit
log.info {
val htlcsIn = spec.htlcs.outgoings().map { it.id }.joinToString(",")
val htlcsOut = spec.htlcs.incomings().map { it.id }.joinToString(",")
"built remote commit number=${remoteCommit.index + 1} toLocalMsat=${spec.toLocal.toLong()} toRemoteMsat=${spec.toRemote.toLong()} htlc_in=$htlcsIn htlc_out=$htlcsOut feeratePerKw=${spec.feerate} txId=${remoteCommitTx.tx.txid} fundingTxId=$fundingTxId"
}
val commitSig = CommitSig(params.channelId, sig, htlcSigs.toList())
val commitment1 = copy(nextRemoteCommit = NextRemoteCommit(commitSig, RemoteCommit(remoteCommit.index + 1, spec, remoteCommitTx.tx.txid, remoteNextPerCommitmentPoint)))
return Pair(commitment1, commitSig)
}
fun receiveCommit(channelKeys: KeyManager.ChannelKeys, params: ChannelParams, changes: CommitmentChanges, commit: CommitSig, log: MDCLogger): Either {
// they sent us a signature for *their* view of *our* next commit tx
// so in terms of rev.hashes and indexes we have:
// ourCommit.index -> our current revocation hash, which is about to become our old revocation hash
// ourCommit.index + 1 -> our next revocation hash, used by *them* to build the sig we've just received, and which
// is about to become our current revocation hash
// ourCommit.index + 2 -> which is about to become our next revocation hash
// we will reply to this sig with our old revocation hash preimage (at index) and our next revocation hash (at index + 1)
// and will increment our index
// check that their signature is valid
// signatures are now optional in the commit message, and will be sent only if the other party is actually
// receiving money i.e its commit tx has one output for them
val spec = CommitmentSpec.reduce(localCommit.spec, changes.localChanges.acked, changes.remoteChanges.proposed)
val localPerCommitmentPoint = channelKeys.commitmentPoint(localCommit.index + 1)
val (localCommitTx, htlcTxs) = Commitments.makeLocalTxs(channelKeys, commitTxNumber = localCommit.index + 1, params.localParams, params.remoteParams, fundingTxIndex = fundingTxIndex, remoteFundingPubKey = remoteFundingPubkey, commitInput, localPerCommitmentPoint = localPerCommitmentPoint, spec)
val sig = Transactions.sign(localCommitTx, channelKeys.fundingKey(fundingTxIndex))
log.info {
val htlcsIn = spec.htlcs.incomings().map { it.id }.joinToString(",")
val htlcsOut = spec.htlcs.outgoings().map { it.id }.joinToString(",")
"built local commit number=${localCommit.index + 1} toLocalMsat=${spec.toLocal.toLong()} toRemoteMsat=${spec.toRemote.toLong()} htlc_in=$htlcsIn htlc_out=$htlcsOut feeratePerKw=${spec.feerate} txId=${localCommitTx.tx.txid} fundingTxId=$fundingTxId"
}
// no need to compute htlc sigs if commit sig doesn't check out
val signedCommitTx = Transactions.addSigs(localCommitTx, channelKeys.fundingPubKey(fundingTxIndex), remoteFundingPubkey, sig, commit.signature)
when (val check = Transactions.checkSpendable(signedCommitTx)) {
is Try.Failure -> {
log.error(check.error) { "remote signature $commit is invalid" }
return Either.Left(InvalidCommitmentSignature(params.channelId, signedCommitTx.tx.txid))
}
else -> {}
}
val sortedHtlcTxs: List = htlcTxs.sortedBy { it.input.outPoint.index }
if (commit.htlcSignatures.size != sortedHtlcTxs.size) {
return Either.Left(HtlcSigCountMismatch(params.channelId, sortedHtlcTxs.size, commit.htlcSignatures.size))
}
val htlcSigs = sortedHtlcTxs.map { Transactions.sign(it, channelKeys.htlcKey.deriveForCommitment(localPerCommitmentPoint), SigHash.SIGHASH_ALL) }
val remoteHtlcPubkey = params.remoteParams.htlcBasepoint.deriveForCommitment(localPerCommitmentPoint)
// combine the sigs to make signed txs
val htlcTxsAndSigs = Triple(sortedHtlcTxs, htlcSigs, commit.htlcSignatures).zipped().map { (htlcTx, localSig, remoteSig) ->
when (htlcTx) {
is HtlcTx.HtlcTimeoutTx -> {
if (Transactions.checkSpendable(Transactions.addSigs(htlcTx, localSig, remoteSig)).isFailure) {
return Either.Left(InvalidHtlcSignature(params.channelId, htlcTx.tx.txid))
}
HtlcTxAndSigs(htlcTx, localSig, remoteSig)
}
is HtlcTx.HtlcSuccessTx -> {
// we can't check that htlc-success tx are spendable because we need the payment preimage; thus we only check the remote sig
// which was created with SIGHASH_SINGLE || SIGHASH_ANYONECANPAY
if (!Transactions.checkSig(htlcTx, remoteSig, remoteHtlcPubkey, SigHash.SIGHASH_SINGLE or SigHash.SIGHASH_ANYONECANPAY)) {
return Either.Left(InvalidHtlcSignature(params.channelId, htlcTx.tx.txid))
}
HtlcTxAndSigs(htlcTx, localSig, remoteSig)
}
}
}
val localCommit1 = LocalCommit(localCommit.index + 1, spec, PublishableTxs(signedCommitTx, htlcTxsAndSigs))
return Either.Right(copy(localCommit = localCommit1))
}
}
/** Subset of Commitments when we want to work with a single, specific commitment. */
data class FullCommitment(
val params: ChannelParams, val changes: CommitmentChanges,
val fundingTxIndex: Long,
val remoteFundingPubkey: PublicKey,
val localFundingStatus: LocalFundingStatus, val remoteFundingStatus: RemoteFundingStatus,
val localCommit: LocalCommit, val remoteCommit: RemoteCommit, val nextRemoteCommit: NextRemoteCommit?
) {
val channelId = params.channelId
val commitInput = localCommit.publishableTxs.commitTx.input
val fundingTxId: ByteVector32 = commitInput.outPoint.txid
val fundingAmount = commitInput.txOut.amount
val localChannelReserve = when {
params.channelFeatures.hasFeature(Feature.ZeroReserveChannels) && !params.localParams.isInitiator -> 0.sat
else -> (fundingAmount / 100).max(params.remoteParams.dustLimit)
}
val remoteChannelReserve = when {
params.channelFeatures.hasFeature(Feature.ZeroReserveChannels) && params.localParams.isInitiator -> 0.sat
else -> (fundingAmount / 100).max(params.localParams.dustLimit)
}
}
data class WaitingForRevocation(val sentAfterLocalCommitIndex: Long)
data class Commitments(
val params: ChannelParams,
val changes: CommitmentChanges,
val active: List,
val inactive: List,
val payments: Map, // for outgoing htlcs, maps to paymentId
val remoteNextCommitInfo: Either, // this one is tricky, it must be kept in sync with Commitment.nextRemoteCommit
val remotePerCommitmentSecrets: ShaChain,
val remoteChannelData: EncryptedChannelData = EncryptedChannelData.empty
) {
init {
require(active.isNotEmpty()) { "there must be at least one active commitment" }
}
val channelId: ByteVector32 = params.channelId
val localNodeId: PublicKey = params.localParams.nodeId
val remoteNodeId: PublicKey = params.remoteParams.nodeId
// Commitment numbers are the same for all active commitments.
val localCommitIndex = active.first().localCommit.index
val remoteCommitIndex = active.first().remoteCommit.index
val nextRemoteCommitIndex = remoteCommitIndex + 1
fun availableBalanceForSend(): MilliSatoshi = active.minOf { it.availableBalanceForSend(params, changes) }
fun availableBalanceForReceive(): MilliSatoshi = active.minOf { it.availableBalanceForReceive(params, changes) }
// We always use the last commitment that was created, to make sure we never go back in time.
val latest = FullCommitment(params, changes, active.first().fundingTxIndex, active.first().remoteFundingPubkey, active.first().localFundingStatus, active.first().remoteFundingStatus, active.first().localCommit, active.first().remoteCommit, active.first().nextRemoteCommit)
val all = buildList {
addAll(active)
addAll(inactive)
}
fun add(commitment: Commitment): Commitments = copy(active = buildList {
add(commitment)
addAll(active)
})
fun isMoreRecent(other: Commitments): Boolean {
return this.localCommitIndex > other.localCommitIndex ||
this.remoteCommitIndex > other.remoteCommitIndex ||
(this.remoteCommitIndex == other.remoteCommitIndex && this.remoteNextCommitInfo.isLeft && other.remoteNextCommitInfo.isRight) ||
this.latest.fundingTxIndex > other.latest.fundingTxIndex
}
// @formatter:off
// HTLCs and pending changes are the same for all active commitments, so we don't need to loop through all of them.
fun isIdle(): Boolean = active.first().isIdle(changes)
fun hasNoPendingHtlcsOrFeeUpdate(): Boolean = active.first().hasNoPendingHtlcsOrFeeUpdate(changes)
fun timedOutOutgoingHtlcs(currentHeight: Long): Set = active.first().timedOutOutgoingHtlcs(currentHeight)
fun almostTimedOutIncomingHtlcs(currentHeight: Long, fulfillSafety: CltvExpiryDelta): Set = active.first().almostTimedOutIncomingHtlcs(currentHeight, fulfillSafety, changes)
fun getOutgoingHtlcCrossSigned(htlcId: Long): UpdateAddHtlc? = active.first().getOutgoingHtlcCrossSigned(htlcId)
fun getIncomingHtlcCrossSigned(htlcId: Long): UpdateAddHtlc? = active.first().getIncomingHtlcCrossSigned(htlcId)
// @formatter:on
fun sendAdd(cmd: ChannelCommand.Htlc.Add, paymentId: UUID, blockHeight: Long): Either> {
val maxExpiry = Channel.MAX_CLTV_EXPIRY_DELTA.toCltvExpiry(blockHeight)
// we don't want to use too high a refund timeout, because our funds will be locked during that time if the payment is never fulfilled
if (cmd.cltvExpiry >= maxExpiry) {
return Either.Left(ExpiryTooBig(channelId, maximum = maxExpiry, actual = cmd.cltvExpiry, blockCount = blockHeight))
}
// even if remote advertises support for 0 msat htlc, we limit ourselves to values strictly positive, hence the max(1 msat)
val htlcMinimum = params.remoteParams.htlcMinimum.coerceAtLeast(1.msat)
if (cmd.amount < htlcMinimum) {
return Either.Left(HtlcValueTooSmall(channelId, minimum = htlcMinimum, actual = cmd.amount))
}
// let's compute the current commitment *as seen by them* with this change taken into account
val add = UpdateAddHtlc(channelId, changes.localNextHtlcId, cmd.amount, cmd.paymentHash, cmd.cltvExpiry, cmd.onion)
// we increment the local htlc index and add an entry to the origins map
val changes1 = changes.addLocalProposal(add).copy(localNextHtlcId = changes.localNextHtlcId + 1)
val payments1 = payments + mapOf(add.id to paymentId)
val failure = active.map { it.canSendAdd(cmd.amount, params, changes1).left }.firstOrNull()
return failure?.let { Either.Left(it) } ?: Either.Right(Pair(copy(changes = changes1, payments = payments1), add))
}
fun receiveAdd(add: UpdateAddHtlc): Either {
if (add.id != changes.remoteNextHtlcId) {
return Either.Left(UnexpectedHtlcId(channelId, expected = changes.remoteNextHtlcId, actual = add.id))
}
// we used to not enforce a strictly positive minimum, hence the max(1 msat)
val htlcMinimum = params.localParams.htlcMinimum.coerceAtLeast(1.msat)
if (add.amountMsat < htlcMinimum) {
return Either.Left(HtlcValueTooSmall(channelId, minimum = htlcMinimum, actual = add.amountMsat))
}
val changes1 = changes.addRemoteProposal(add).copy(remoteNextHtlcId = changes.remoteNextHtlcId + 1)
val failure = active.map { it.canReceiveAdd(add.amountMsat, params, changes1).left }.firstOrNull()
return failure?.let { Either.Left(it) } ?: Either.Right(copy(changes = changes1))
}
fun sendFulfill(cmd: ChannelCommand.Htlc.Settlement.Fulfill): Either> {
val htlc = getIncomingHtlcCrossSigned(cmd.id) ?: return Either.Left(UnknownHtlcId(channelId, cmd.id))
return when {
// we have already sent a fail/fulfill for this htlc
CommitmentChanges.alreadyProposed(changes.localChanges.proposed, htlc.id) -> Either.Left(UnknownHtlcId(channelId, cmd.id))
htlc.paymentHash.contentEquals(sha256(cmd.r)) -> {
val fulfill = UpdateFulfillHtlc(channelId, cmd.id, cmd.r)
Either.Right(Pair(copy(changes = changes.addLocalProposal(fulfill)), fulfill))
}
else -> Either.Left(InvalidHtlcPreimage(channelId, cmd.id))
}
}
fun receiveFulfill(fulfill: UpdateFulfillHtlc): Either> {
val htlc = getOutgoingHtlcCrossSigned(fulfill.id) ?: return Either.Left(UnknownHtlcId(channelId, fulfill.id))
val paymentId = payments[fulfill.id] ?: return Either.Left(UnknownHtlcId(channelId, fulfill.id))
return when {
htlc.paymentHash.contentEquals(sha256(fulfill.paymentPreimage)) -> Either.Right(Triple(copy(changes = changes.addRemoteProposal(fulfill)), paymentId, htlc))
else -> Either.Left(InvalidHtlcPreimage(channelId, fulfill.id))
}
}
fun sendFail(cmd: ChannelCommand.Htlc.Settlement.Fail, nodeSecret: PrivateKey): Either> {
val htlc = getIncomingHtlcCrossSigned(cmd.id) ?: return Either.Left(UnknownHtlcId(channelId, cmd.id))
return when {
// we have already sent a fail/fulfill for this htlc
CommitmentChanges.alreadyProposed(changes.localChanges.proposed, htlc.id) -> Either.Left(UnknownHtlcId(channelId, cmd.id))
else -> {
when (val result = OutgoingPaymentPacket.buildHtlcFailure(nodeSecret, htlc.paymentHash, htlc.onionRoutingPacket, cmd.reason)) {
is Either.Right -> {
val fail = UpdateFailHtlc(channelId, cmd.id, result.value)
Either.Right(Pair(copy(changes = changes.addLocalProposal(fail)), fail))
}
is Either.Left -> Either.Left(CannotExtractSharedSecret(channelId, htlc))
}
}
}
}
fun sendFailMalformed(cmd: ChannelCommand.Htlc.Settlement.FailMalformed): Either> {
// BADONION bit must be set in failure_code
if ((cmd.failureCode and FailureMessage.BADONION) == 0) return Either.Left(InvalidFailureCode(channelId))
val htlc = getIncomingHtlcCrossSigned(cmd.id) ?: return Either.Left(UnknownHtlcId(channelId, cmd.id))
return when {
// we have already sent a fail/fulfill for this htlc
CommitmentChanges.alreadyProposed(changes.localChanges.proposed, htlc.id) -> Either.Left(UnknownHtlcId(channelId, cmd.id))
else -> {
val fail = UpdateFailMalformedHtlc(channelId, cmd.id, cmd.onionHash, cmd.failureCode)
Either.Right(Pair(copy(changes = changes.addLocalProposal(fail)), fail))
}
}
}
fun receiveFail(fail: UpdateFailHtlc): Either> {
val htlc = getOutgoingHtlcCrossSigned(fail.id) ?: return Either.Left(UnknownHtlcId(channelId, fail.id))
val paymentId = payments[fail.id] ?: return Either.Left(UnknownHtlcId(channelId, fail.id))
return Either.Right(Triple(copy(changes = changes.addRemoteProposal(fail)), paymentId, htlc))
}
fun receiveFailMalformed(fail: UpdateFailMalformedHtlc): Either> {
// A receiving node MUST fail the channel if the BADONION bit in failure_code is not set for update_fail_malformed_htlc.
if ((fail.failureCode and FailureMessage.BADONION) == 0) return Either.Left(InvalidFailureCode(channelId))
val htlc = getOutgoingHtlcCrossSigned(fail.id) ?: return Either.Left(UnknownHtlcId(channelId, fail.id))
val paymentId = payments[fail.id] ?: return Either.Left(UnknownHtlcId(channelId, fail.id))
return Either.Right(Triple(copy(changes = changes.addRemoteProposal(fail)), paymentId, htlc))
}
fun sendFee(cmd: ChannelCommand.Commitment.UpdateFee): Either> {
if (!params.localParams.isInitiator) return Either.Left(NonInitiatorCannotSendUpdateFee(channelId))
// let's compute the current commitment *as seen by them* with this change taken into account
val fee = UpdateFee(channelId, cmd.feerate)
// update_fee replace each other, so we can remove previous ones
val changes1 = changes.copy(localChanges = changes.localChanges.copy(proposed = changes.localChanges.proposed.filterNot { it is UpdateFee } + fee))
val failure = active.map { it.canSendFee(params, changes1).left }.firstOrNull()
return failure?.let { Either.Left(it) } ?: Either.Right(Pair(copy(changes = changes1), fee))
}
fun receiveFee(fee: UpdateFee, feerateTolerance: FeerateTolerance): Either {
if (params.localParams.isInitiator) return Either.Left(NonInitiatorCannotSendUpdateFee(channelId))
if (fee.feeratePerKw < FeeratePerKw.MinimumFeeratePerKw) return Either.Left(FeerateTooSmall(channelId, remoteFeeratePerKw = fee.feeratePerKw))
if (Helpers.isFeeDiffTooHigh(FeeratePerKw.CommitmentFeerate, fee.feeratePerKw, feerateTolerance)) return Either.Left(FeerateTooDifferent(channelId, FeeratePerKw.CommitmentFeerate, fee.feeratePerKw))
val changes1 = changes.copy(remoteChanges = changes.remoteChanges.copy(proposed = changes.remoteChanges.proposed.filterNot { it is UpdateFee } + fee))
val failure = active.map { it.canReceiveFee(params, changes1).left }.firstOrNull()
return failure?.let { Either.Left(it) } ?: Either.Right(copy(changes = changes1))
}
fun sendCommit(channelKeys: KeyManager.ChannelKeys, log: MDCLogger): Either>> {
val remoteNextPerCommitmentPoint = remoteNextCommitInfo.right ?: return Either.Left(CannotSignBeforeRevocation(channelId))
if (!changes.localHasChanges()) return Either.Left(CannotSignWithoutChanges(channelId))
val (active1, sigs) = active.map { it.sendCommit(channelKeys, params, changes, remoteNextPerCommitmentPoint, log) }.unzip()
val commitments1 = copy(
active = active1,
remoteNextCommitInfo = Either.Left(WaitingForRevocation(localCommitIndex)),
changes = changes.copy(
localChanges = changes.localChanges.copy(proposed = emptyList(), signed = changes.localChanges.proposed),
remoteChanges = changes.remoteChanges.copy(acked = emptyList(), signed = changes.remoteChanges.acked)
)
)
val sigs1 = if (sigs.size > 1) {
sigs.map { sig ->
sig.copy(tlvStream = sig.tlvStream.copy(records = buildSet {
addAll(sig.tlvStream.records)
add(CommitSigTlv.Batch(sigs.size))
}))
}
} else sigs
return Either.Right(Pair(commitments1, sigs1))
}
fun receiveCommit(commits: List, channelKeys: KeyManager.ChannelKeys, log: MDCLogger): Either> {
// We may receive more commit_sig than the number of active commitments, because there can be a race where we send splice_locked
// while our peer is sending us a batch of commit_sig. When that happens, we simply need to discard the commit_sig that belong
// to commitments we deactivated.
if (commits.size < active.size) {
return Either.Left(CommitSigCountMismatch(channelId, active.size, commits.size))
}
// Signatures are sent in order (most recent first), calling `zip` will drop trailing sigs that are for deactivated/pruned commitments.
val active1 = active.zip(commits).map {
when (val commitment1 = it.first.receiveCommit(channelKeys, params, changes, it.second, log)) {
is Either.Left -> return Either.Left(commitment1.value)
is Either.Right -> commitment1.value
}
}
// we will send our revocation preimage + our next revocation hash
val localPerCommitmentSecret = channelKeys.commitmentSecret(localCommitIndex)
val localNextPerCommitmentPoint = channelKeys.commitmentPoint(localCommitIndex + 2)
val revocation = RevokeAndAck(channelId, localPerCommitmentSecret, localNextPerCommitmentPoint)
val commitments1 = copy(
active = active1,
changes = changes.copy(
localChanges = changes.localChanges.copy(acked = emptyList()),
remoteChanges = changes.remoteChanges.copy(proposed = emptyList(), acked = changes.remoteChanges.acked + changes.remoteChanges.proposed)
),
remoteChannelData = commits.last().channelData // the last message is the most recent
)
return Either.Right(Pair(commitments1, revocation))
}
fun receiveRevocation(revocation: RevokeAndAck): Either>> {
if (remoteNextCommitInfo.isRight) return Either.Left(UnexpectedRevocation(channelId))
// Since htlcs are shared across all commitments, we generate the actions only once based on the first commitment.
val remoteCommit = active.first().remoteCommit
if (revocation.perCommitmentSecret.publicKey() != remoteCommit.remotePerCommitmentPoint) return Either.Left(InvalidRevocation(channelId))
// the outgoing following htlcs have been completed (fulfilled or failed) when we received this revocation
// they have been removed from both local and remote commitment
// since fulfill/fail are sent by remote, they are (1) signed by them, (2) revoked by us, (3) signed by us, (4) revoked by them
val completedOutgoingHtlcs = changes.remoteChanges.signed.mapNotNull {
when (it) {
is UpdateFulfillHtlc -> it.id
is UpdateFailHtlc -> it.id
is UpdateFailMalformedHtlc -> it.id
else -> null
}
}
// we remove the newly completed htlcs from the payments map
val payments1 = payments - completedOutgoingHtlcs.toSet()
val actions = mutableListOf()
changes.remoteChanges.signed.forEach {
when (it) {
is UpdateAddHtlc -> actions += ChannelAction.ProcessIncomingHtlc(it)
is UpdateFailHtlc -> {
val paymentId = payments[it.id]
val add = remoteCommit.spec.findIncomingHtlcById(it.id)?.add
if (paymentId != null && add != null) {
actions += ChannelAction.ProcessCmdRes.AddSettledFail(paymentId, add, ChannelAction.HtlcResult.Fail.RemoteFail(it))
}
}
is UpdateFailMalformedHtlc -> {
val paymentId = payments[it.id]
val add = remoteCommit.spec.findIncomingHtlcById(it.id)?.add
if (paymentId != null && add != null) {
actions += ChannelAction.ProcessCmdRes.AddSettledFail(paymentId, add, ChannelAction.HtlcResult.Fail.RemoteFailMalformed(it))
}
}
else -> Unit
}
}
val active1 = active.map { it.copy(remoteCommit = it.nextRemoteCommit!!.commit, nextRemoteCommit = null) }
val commitments1 = this.copy(
active = active1,
changes = changes.copy(
localChanges = changes.localChanges.copy(signed = emptyList(), acked = changes.localChanges.acked + changes.localChanges.signed),
remoteChanges = changes.remoteChanges.copy(signed = emptyList()),
),
remoteNextCommitInfo = Either.Right(revocation.nextPerCommitmentPoint),
remotePerCommitmentSecrets = remotePerCommitmentSecrets.addHash(revocation.perCommitmentSecret.value, 0xFFFFFFFFFFFFL - remoteCommitIndex),
payments = payments1,
remoteChannelData = revocation.channelData
)
return Either.Right(Pair(commitments1, actions.toList()))
}
private fun ChannelContext.updateFundingStatus(fundingTxId: ByteVector32, updateMethod: (Commitment, Long) -> Commitment): Either> {
return when (val c = all.find { it.fundingTxId == fundingTxId }) {
is Commitment -> {
val commitments1 = copy(
active = active.map { updateMethod(it, c.fundingTxIndex) },
inactive = inactive.map { updateMethod(it, c.fundingTxIndex) },
)
val commitment = commitments1.all.find { it.fundingTxId == fundingTxId }!! // NB: this commitment might be pruned at the next line
val commitments2 = commitments1.run { deactivateCommitments() }.run { pruneCommitments() }
logger.info { "commitments active=${commitments2.active.map { it.fundingTxIndex }} inactive=${commitments2.inactive.map { it.fundingTxIndex }}" }
Either.Right(Pair(commitments2, commitment))
}
else -> {
logger.warning { "fundingTxId=$fundingTxId doesn't match any of our funding txs" }
Either.Left(this@Commitments)
}
}
}
fun ChannelContext.updateLocalFundingSigned(fundingTx: FullySignedSharedTransaction): Either> =
updateFundingStatus(fundingTx.txId) { c: Commitment, _: Long ->
if (c.fundingTxId == fundingTx.txId) {
when (c.localFundingStatus) {
is LocalFundingStatus.UnconfirmedFundingTx -> {
logger.debug { "setting localFundingStatus fully signed for fundingTxId=${fundingTx.txId}" }
c.copy(localFundingStatus = c.localFundingStatus.copy(sharedTx = fundingTx))
}
is LocalFundingStatus.ConfirmedFundingTx -> c
}
} else c
}
fun ChannelContext.updateLocalFundingConfirmed(fundingTx: Transaction): Either> =
updateFundingStatus(fundingTx.txid) { c: Commitment, _: Long ->
if (c.fundingTxId == fundingTx.txid) {
when (c.localFundingStatus) {
is LocalFundingStatus.UnconfirmedFundingTx -> {
logger.debug { "setting localFundingStatus confirmed for fundingTxId=${fundingTx.txid}" }
c.copy(localFundingStatus = LocalFundingStatus.ConfirmedFundingTx(fundingTx, c.localFundingStatus.sharedTx.tx.fees, c.localFundingStatus.sharedTx.localSigs))
}
is LocalFundingStatus.ConfirmedFundingTx -> c
}
} else c
}
fun ChannelContext.updateRemoteFundingStatus(fundingTxId: ByteVector32): Either> =
updateFundingStatus(fundingTxId) { c: Commitment, fundingTxIndex: Long ->
// all funding older than this one are considered locked
if (c.fundingTxId == fundingTxId || c.fundingTxIndex < fundingTxIndex) {
logger.debug { "setting remoteFundingStatus=${RemoteFundingStatus.Locked::class.simpleName} for fundingTxId=$fundingTxId" }
c.copy(remoteFundingStatus = RemoteFundingStatus.Locked)
} else c
}
/**
* Return the most recent commitment locked by both sides.
*/
fun ChannelContext.lastLocked(): Commitment? {
// When a commitment is locked, it implicitly locks all previous commitments.
// This ensures that we only have to send splice_locked for the latest commitment instead of sending it for every commitment.
// A side-effect is that previous commitments that are implicitly locked don't necessarily have their status correctly set.
// That's why we look at locked commitments separately and then select the one with the oldest fundingTxIndex.
val lastLocalLocked = active.find { staticParams.useZeroConf || it.localFundingStatus is LocalFundingStatus.ConfirmedFundingTx }
val lastRemoteLocked = active.find { it.remoteFundingStatus == RemoteFundingStatus.Locked }
return when {
// We select the locked commitment with the smaller value for fundingTxIndex, but both have to be defined.
// If both have the same fundingTxIndex, they must actually be the same commitment, because:
// - we only allow RBF attempts when we're not using zero-conf
// - transactions with the same fundingTxIndex double-spend each other, so only one of them can confirm
// - we don't allow creating a splice on top of an unconfirmed transaction that has RBF attempts (because it
// would become invalid if another of the RBF attempts end up being confirmed)
lastLocalLocked != null && lastRemoteLocked != null -> listOf(lastLocalLocked, lastRemoteLocked).minByOrNull { it.fundingTxIndex }
// Special case for the initial funding tx, we only require a local lock because channel_ready doesn't explicitly reference a funding tx.
lastLocalLocked != null && lastLocalLocked.fundingTxIndex == 0L -> lastLocalLocked
else -> null
}
}
/**
* Commitments are considered inactive when they have been superseded by a newer commitment, but can still potentially
* end up on-chain. This is a consequence of using zero-conf. Inactive commitments will be cleaned up by
* [pruneCommitments], when the next funding tx confirms.
*/
private fun ChannelContext.deactivateCommitments(): Commitments = when (val commitment = lastLocked()) {
is Commitment -> {
// all commitments older than this one are inactive
val inactive1 = active.filter { it.fundingTxId != commitment.fundingTxId && it.fundingTxIndex <= commitment.fundingTxIndex }
inactive1.forEach { logger.info { "deactivating commitment fundingTxIndex=${it.fundingTxIndex} fundingTxId=${it.fundingTxId}" } }
copy(
active = active - inactive1.toSet(),
inactive = inactive1 + inactive.toSet()
)
}
else -> this@Commitments
}
/**
* We can prune commitments in two cases:
* - their funding tx has been permanently double-spent by the funding tx of a concurrent commitment (happens when using RBF)
* - their funding tx has been permanently spent by a splice tx
*/
private fun ChannelContext.pruneCommitments(): Commitments {
return when (val lastConfirmed = all.find { it.localFundingStatus is LocalFundingStatus.ConfirmedFundingTx }) {
null -> this@Commitments
else -> {
// We can prune all other commitments with the same or lower funding index.
// NB: we cannot prune active commitments, even if we know that they have been double-spent, because our peer may not yet
// be aware of it, and will expect us to send commit_sig.
val pruned = inactive.filter { it.fundingTxId != lastConfirmed.fundingTxId && it.fundingTxIndex <= lastConfirmed.fundingTxIndex }
pruned.forEach { logger.info { "pruning commitment fundingTxIndex=${it.fundingTxIndex} fundingTxId=${it.fundingTxId}" } }
copy(inactive = inactive - pruned.toSet())
}
}
}
/**
* Find the corresponding commitment, based on a spending transaction.
*
* @param spendingTx A transaction that may spend a current or former funding tx
*/
fun resolveCommitment(spendingTx: Transaction): Commitment? {
return all.find { commitment -> spendingTx.txIn.map { it.outPoint }.contains(commitment.commitInput.outPoint) }
}
companion object {
val ANCHOR_AMOUNT = 330.sat
const val COMMIT_WEIGHT = 1124
const val HTLC_OUTPUT_WEIGHT = 172
const val HTLC_TIMEOUT_WEIGHT = 666
const val HTLC_SUCCESS_WEIGHT = 706
fun makeLocalTxs(
channelKeys: KeyManager.ChannelKeys,
commitTxNumber: Long,
localParams: LocalParams,
remoteParams: RemoteParams,
fundingTxIndex: Long,
remoteFundingPubKey: PublicKey,
commitmentInput: Transactions.InputInfo,
localPerCommitmentPoint: PublicKey,
spec: CommitmentSpec
): Pair> {
val localDelayedPaymentPubkey = channelKeys.delayedPaymentBasepoint.deriveForCommitment(localPerCommitmentPoint)
val localHtlcPubkey = channelKeys.htlcBasepoint.deriveForCommitment(localPerCommitmentPoint)
val remotePaymentPubkey = remoteParams.paymentBasepoint
val remoteHtlcPubkey = remoteParams.htlcBasepoint.deriveForCommitment(localPerCommitmentPoint)
val localRevocationPubkey = remoteParams.revocationBasepoint.deriveForRevocation(localPerCommitmentPoint)
val localPaymentBasepoint = channelKeys.paymentBasepoint
val outputs = makeCommitTxOutputs(
channelKeys.fundingPubKey(fundingTxIndex),
remoteFundingPubKey,
localParams.isInitiator,
localParams.dustLimit,
localRevocationPubkey,
remoteParams.toSelfDelay,
localDelayedPaymentPubkey,
remotePaymentPubkey,
localHtlcPubkey,
remoteHtlcPubkey,
spec
)
val commitTx = Transactions.makeCommitTx(commitmentInput, commitTxNumber, localPaymentBasepoint, remoteParams.paymentBasepoint, localParams.isInitiator, outputs)
val htlcTxs = Transactions.makeHtlcTxs(commitTx.tx, localParams.dustLimit, localRevocationPubkey, remoteParams.toSelfDelay, localDelayedPaymentPubkey, spec.feerate, outputs)
return Pair(commitTx, htlcTxs)
}
fun makeRemoteTxs(
channelKeys: KeyManager.ChannelKeys,
commitTxNumber: Long,
localParams: LocalParams,
remoteParams: RemoteParams,
fundingTxIndex: Long,
remoteFundingPubKey: PublicKey,
commitmentInput: Transactions.InputInfo,
remotePerCommitmentPoint: PublicKey,
spec: CommitmentSpec
): Pair> {
val localPaymentPubkey = channelKeys.paymentBasepoint
val localHtlcPubkey = channelKeys.htlcBasepoint.deriveForCommitment(remotePerCommitmentPoint)
val remoteDelayedPaymentPubkey = remoteParams.delayedPaymentBasepoint.deriveForCommitment(remotePerCommitmentPoint)
val remoteHtlcPubkey = remoteParams.htlcBasepoint.deriveForCommitment(remotePerCommitmentPoint)
val remoteRevocationPubkey = channelKeys.revocationBasepoint.deriveForRevocation(remotePerCommitmentPoint)
val outputs = makeCommitTxOutputs(
remoteFundingPubKey,
channelKeys.fundingPubKey(fundingTxIndex),
!localParams.isInitiator,
remoteParams.dustLimit,
remoteRevocationPubkey,
localParams.toSelfDelay,
remoteDelayedPaymentPubkey,
localPaymentPubkey,
remoteHtlcPubkey,
localHtlcPubkey,
spec
)
// NB: we are creating the remote commit tx, so local/remote parameters are inverted.
val commitTx = Transactions.makeCommitTx(commitmentInput, commitTxNumber, remoteParams.paymentBasepoint, localPaymentPubkey, !localParams.isInitiator, outputs)
val htlcTxs = Transactions.makeHtlcTxs(commitTx.tx, remoteParams.dustLimit, remoteRevocationPubkey, localParams.toSelfDelay, remoteDelayedPaymentPubkey, spec.feerate, outputs)
return Pair(commitTx, htlcTxs)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy