commonMain.fr.acinq.lightning.wire.LightningMessages.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of lightning-kmp-jvm Show documentation
Show all versions of lightning-kmp-jvm Show documentation
A Kotlin Multiplatform implementation of the Lightning Network
package fr.acinq.lightning.wire
import fr.acinq.bitcoin.*
import fr.acinq.bitcoin.crypto.musig2.IndividualNonce
import fr.acinq.bitcoin.io.ByteArrayInput
import fr.acinq.bitcoin.io.ByteArrayOutput
import fr.acinq.bitcoin.io.Input
import fr.acinq.bitcoin.io.Output
import fr.acinq.lightning.*
import fr.acinq.lightning.blockchain.fee.FeeratePerKw
import fr.acinq.lightning.channel.ChannelType
import fr.acinq.lightning.channel.Origin
import fr.acinq.lightning.logging.*
import fr.acinq.lightning.router.Announcements
import fr.acinq.lightning.utils.*
import fr.acinq.secp256k1.Hex
import kotlin.math.max
import kotlin.math.min
interface LightningMessage {
/**
* Once decrypted, lightning messages start with a 2-byte type that uniquely identifies the kind of message received.
* See https://github.com/lightningnetwork/lightning-rfc/blob/master/01-messaging.md#lightning-message-format.
*/
val type: Long
fun write(out: Output)
fun write(): ByteArray {
val out = ByteArrayOutput()
write(out)
return out.toByteArray()
}
companion object {
/**
* @param input a single, complete message (typically received over the transport layer).
* There is a very strong assumption that framing has been taken care of and that there are no missing or extra bytes.
* Whatever we don't read will simply be ignored, as per the BOLTs.
*/
fun decode(input: ByteArray): LightningMessage {
val stream = ByteArrayInput(input)
val code = LightningCodecs.u16(stream)
return when (code.toLong()) {
Init.type -> Init.read(stream)
ChannelReestablish.type -> ChannelReestablish.read(stream)
Warning.type -> Warning.read(stream)
Error.type -> Error.read(stream)
Ping.type -> Ping.read(stream)
Pong.type -> Pong.read(stream)
OpenDualFundedChannel.type -> OpenDualFundedChannel.read(stream)
AcceptDualFundedChannel.type -> AcceptDualFundedChannel.read(stream)
FundingCreated.type -> FundingCreated.read(stream)
FundingSigned.type -> FundingSigned.read(stream)
ChannelReady.type -> ChannelReady.read(stream)
TxAddInput.type -> TxAddInput.read(stream)
TxAddOutput.type -> TxAddOutput.read(stream)
TxRemoveInput.type -> TxRemoveInput.read(stream)
TxRemoveOutput.type -> TxRemoveOutput.read(stream)
TxComplete.type -> TxComplete.read(stream)
TxSignatures.type -> TxSignatures.read(stream)
TxInitRbf.type -> TxInitRbf.read(stream)
TxAckRbf.type -> TxAckRbf.read(stream)
TxAbort.type -> TxAbort.read(stream)
CommitSig.type -> CommitSig.read(stream)
RevokeAndAck.type -> RevokeAndAck.read(stream)
UpdateAddHtlc.type -> UpdateAddHtlc.read(stream)
UpdateFailHtlc.type -> UpdateFailHtlc.read(stream)
UpdateFailMalformedHtlc.type -> UpdateFailMalformedHtlc.read(stream)
UpdateFulfillHtlc.type -> UpdateFulfillHtlc.read(stream)
UpdateFee.type -> UpdateFee.read(stream)
AnnouncementSignatures.type -> AnnouncementSignatures.read(stream)
ChannelAnnouncement.type -> ChannelAnnouncement.read(stream)
ChannelUpdate.type -> ChannelUpdate.read(stream)
Shutdown.type -> Shutdown.read(stream)
ClosingSigned.type -> ClosingSigned.read(stream)
OnionMessage.type -> OnionMessage.read(stream)
PayToOpenRequest.type -> PayToOpenRequest.read(stream)
PayToOpenResponse.type -> PayToOpenResponse.read(stream)
FCMToken.type -> FCMToken.read(stream)
UnsetFCMToken.type -> UnsetFCMToken
PhoenixAndroidLegacyInfo.type -> PhoenixAndroidLegacyInfo.read(stream)
PleaseOpenChannel.type -> PleaseOpenChannel.read(stream)
CurrentFeeCredit.type -> CurrentFeeCredit.read(stream)
AutoLiquidityParams.type -> AutoLiquidityParams.read(stream)
Stfu.type -> Stfu.read(stream)
SpliceInit.type -> SpliceInit.read(stream)
SpliceAck.type -> SpliceAck.read(stream)
SpliceLocked.type -> SpliceLocked.read(stream)
else -> UnknownMessage(code.toLong())
}
}
fun encode(input: LightningMessage, out: Output) {
LightningCodecs.writeU16(input.type.toInt(), out)
input.write(out)
}
fun encode(input: LightningMessage): ByteArray {
val out = ByteArrayOutput()
encode(input, out)
return out.toByteArray()
}
}
}
interface LightningMessageReader {
fun read(input: Input): T
fun read(bytes: ByteArray): T = read(ByteArrayInput(bytes))
fun read(hex: String): T = read(Hex.decode(hex))
}
interface HtlcMessage : LightningMessage
interface SetupMessage : LightningMessage
interface RoutingMessage : LightningMessage
interface AnnouncementMessage : RoutingMessage
interface HasTimestamp : LightningMessage {
val timestampSeconds: Long
}
interface UpdateMessage : LightningMessage, ForbiddenMessageDuringSplice
interface HtlcSettlementMessage : UpdateMessage {
val id: Long
}
sealed class InteractiveTxMessage : LightningMessage
sealed class InteractiveTxConstructionMessage : InteractiveTxMessage()
interface HasTemporaryChannelId : LightningMessage {
val temporaryChannelId: ByteVector32
}
interface HasChannelId : LightningMessage {
val channelId: ByteVector32
}
interface HasSerialId : LightningMessage {
val serialId: Long
}
interface HasChainHash : LightningMessage {
val chainHash: BlockHash
}
interface ForbiddenMessageDuringSplice : LightningMessage
data class EncryptedChannelData(val data: ByteVector) {
/** We don't want to log the encrypted channel backups, they take a lot of space. We only keep the first bytes to help correlate mobile/server backups. */
override fun toString(): String {
val bytes = data.take(min(data.size(), 10))
return if (bytes.isEmpty()) "" else "$bytes (truncated)"
}
fun isEmpty(): Boolean = data.isEmpty()
companion object {
val empty: EncryptedChannelData = EncryptedChannelData(ByteVector.empty)
}
}
interface HasEncryptedChannelData : HasChannelId {
val channelData: EncryptedChannelData
fun withNonEmptyChannelData(ecd: EncryptedChannelData): HasEncryptedChannelData
fun withChannelData(data: ByteVector, logger: MDCLogger? = null): HasEncryptedChannelData = withChannelData(EncryptedChannelData(data), logger)
fun withChannelData(ecd: EncryptedChannelData, logger: MDCLogger? = null): HasEncryptedChannelData {
// The maximum size for lightning messages is 65535 bytes, so we only include a channel backup when we have enough space available.
// Our peer needs to be able to send the backup back in their channel_reestablish message, and we don't know the size of that message.
// We use an upper bound of 60000 bytes, hoping that it will be enough to make it work in all cases.
val sizeWithoutBackup = write().size
return when {
ecd.isEmpty() -> this
ecd.data.size() + sizeWithoutBackup <= 60_000 -> withNonEmptyChannelData(ecd)
else -> {
logger?.warning { "c:${this.channelId} could not include encrypted backup in ${this::class.simpleName}: too large (${ecd.data.size()} bytes)" }
this
}
}
}
}
interface ChannelMessage
data class Init(val features: Features, val tlvs: TlvStream = TlvStream.empty()) : SetupMessage {
val networks = tlvs.get()?.chainHashes ?: listOf()
val liquidityRates = tlvs.get()?.leaseRates ?: listOf()
constructor(features: Features, chainHashs: List, liquidityRates: List) : this(
features,
TlvStream(
setOfNotNull(
if (chainHashs.isNotEmpty()) InitTlv.Networks(chainHashs) else null,
if (liquidityRates.isNotEmpty()) InitTlv.LiquidityAdsRates(liquidityRates) else null,
)
)
)
override val type: Long get() = Init.type
override fun write(out: Output) {
LightningCodecs.writeU16(0, out)
features.toByteArray().let {
LightningCodecs.writeU16(it.size, out)
LightningCodecs.writeBytes(it, out)
}
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 16
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
InitTlv.Networks.tag to InitTlv.Networks.Companion as TlvValueReader,
InitTlv.LiquidityAdsRates.tag to InitTlv.LiquidityAdsRates.Companion as TlvValueReader,
InitTlv.PhoenixAndroidLegacyNodeId.tag to InitTlv.PhoenixAndroidLegacyNodeId.Companion as TlvValueReader,
)
override fun read(input: Input): Init {
val gflen = LightningCodecs.u16(input)
val globalFeatures = LightningCodecs.bytes(input, gflen)
val lflen = LightningCodecs.u16(input)
val localFeatures = LightningCodecs.bytes(input, lflen)
val len = max(gflen, lflen)
// merge features together
val features = Features(ByteVector(globalFeatures.leftPaddedCopyOf(len).or(localFeatures.leftPaddedCopyOf(len))))
val tlvs = TlvStreamSerializer(false, readers).read(input)
return Init(features, tlvs)
}
}
}
data class Warning(override val channelId: ByteVector32, val data: ByteVector) : SetupMessage, HasChannelId {
constructor(channelId: ByteVector32, message: String?) : this(channelId, ByteVector(message?.encodeToByteArray() ?: ByteArray(0)))
constructor(message: String?) : this(ByteVector32.Zeroes, message)
fun toAscii(): String = data.toByteArray().decodeToString()
override val type: Long get() = Warning.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU16(data.size(), out)
LightningCodecs.writeBytes(data, out)
}
companion object : LightningMessageReader {
const val type: Long = 1
override fun read(input: Input): Warning {
return Warning(
LightningCodecs.bytes(input, 32).toByteVector32(),
LightningCodecs.bytes(input, LightningCodecs.u16(input)).toByteVector()
)
}
}
}
data class Error(override val channelId: ByteVector32, val data: ByteVector) : SetupMessage, HasChannelId {
constructor(channelId: ByteVector32, message: String?) : this(channelId, ByteVector(message?.encodeToByteArray() ?: ByteArray(0)))
fun toAscii(): String = data.toByteArray().decodeToString()
override val type: Long get() = Error.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU16(data.size(), out)
LightningCodecs.writeBytes(data, out)
}
companion object : LightningMessageReader {
const val type: Long = 17
override fun read(input: Input): Error {
return Error(
LightningCodecs.bytes(input, 32).toByteVector32(),
LightningCodecs.bytes(input, LightningCodecs.u16(input)).toByteVector()
)
}
}
}
data class Ping(val pongLength: Int, val data: ByteVector) : SetupMessage {
override val type: Long get() = Ping.type
override fun write(out: Output) {
LightningCodecs.writeU16(pongLength, out)
LightningCodecs.writeU16(data.size(), out)
LightningCodecs.writeBytes(data, out)
}
companion object : LightningMessageReader {
const val type: Long = 18
override fun read(input: Input): Ping {
return Ping(LightningCodecs.u16(input), LightningCodecs.bytes(input, LightningCodecs.u16(input)).toByteVector())
}
}
}
data class Pong(val data: ByteVector) : SetupMessage {
override val type: Long get() = Pong.type
override fun write(out: Output) {
LightningCodecs.writeU16(data.size(), out)
LightningCodecs.writeBytes(data, out)
}
companion object : LightningMessageReader {
const val type: Long = 19
override fun read(input: Input): Pong {
return Pong(LightningCodecs.bytes(input, LightningCodecs.u16(input)).toByteVector())
}
}
}
data class TxAddInput(
override val channelId: ByteVector32,
override val serialId: Long,
val previousTx: Transaction?,
val previousTxOutput: Long,
val sequence: UInt,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxConstructionMessage(), HasChannelId, HasSerialId {
constructor(channelId: ByteVector32, serialId: Long, sharedInput: OutPoint, sequence: UInt) : this(
channelId,
serialId,
null,
sharedInput.index,
sequence,
TlvStream(TxAddInputTlv.SharedInputTxId(sharedInput.txid))
)
override val type: Long get() = TxAddInput.type
val sharedInput: OutPoint? = tlvs.get()?.let { OutPoint(it.txId, previousTxOutput) }
val swapInParamsLegacy = tlvs.get()
val swapInParams = tlvs.get()
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
LightningCodecs.writeU64(serialId, out)
when (previousTx) {
null -> LightningCodecs.writeU16(0, out)
else -> {
val encodedTx = Transaction.write(previousTx)
LightningCodecs.writeU16(encodedTx.size, out)
LightningCodecs.writeBytes(encodedTx, out)
}
}
LightningCodecs.writeU32(previousTxOutput.toInt(), out)
LightningCodecs.writeU32(sequence.toInt(), out)
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 66
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
TxAddInputTlv.SharedInputTxId.tag to TxAddInputTlv.SharedInputTxId.Companion as TlvValueReader,
TxAddInputTlv.SwapInParamsLegacy.tag to TxAddInputTlv.SwapInParamsLegacy.Companion as TlvValueReader,
TxAddInputTlv.SwapInParams.tag to TxAddInputTlv.SwapInParams.Companion as TlvValueReader,
)
override fun read(input: Input): TxAddInput = TxAddInput(
LightningCodecs.bytes(input, 32).byteVector32(),
LightningCodecs.u64(input),
when (val txSize = LightningCodecs.u16(input)) {
0 -> null
else -> Transaction.read(LightningCodecs.bytes(input, txSize))
},
LightningCodecs.u32(input).toLong(),
LightningCodecs.u32(input).toUInt(),
TlvStreamSerializer(false, readers).read(input)
)
}
}
data class TxAddOutput(
override val channelId: ByteVector32,
override val serialId: Long,
val amount: Satoshi,
val pubkeyScript: ByteVector,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxConstructionMessage(), HasChannelId, HasSerialId {
override val type: Long get() = TxAddOutput.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
LightningCodecs.writeU64(serialId, out)
LightningCodecs.writeU64(amount.toLong(), out)
LightningCodecs.writeU16(pubkeyScript.size(), out)
LightningCodecs.writeBytes(pubkeyScript, out)
}
companion object : LightningMessageReader {
const val type: Long = 67
override fun read(input: Input): TxAddOutput = TxAddOutput(
LightningCodecs.bytes(input, 32).byteVector32(),
LightningCodecs.u64(input),
LightningCodecs.u64(input).sat,
LightningCodecs.bytes(input, LightningCodecs.u16(input)).byteVector(),
)
}
}
data class TxRemoveInput(
override val channelId: ByteVector32,
override val serialId: Long,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxConstructionMessage(), HasChannelId, HasSerialId {
override val type: Long get() = TxRemoveInput.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
LightningCodecs.writeU64(serialId, out)
}
companion object : LightningMessageReader {
const val type: Long = 68
override fun read(input: Input): TxRemoveInput = TxRemoveInput(
LightningCodecs.bytes(input, 32).byteVector32(),
LightningCodecs.u64(input),
)
}
}
data class TxRemoveOutput(
override val channelId: ByteVector32,
override val serialId: Long,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxConstructionMessage(), HasChannelId, HasSerialId {
override val type: Long get() = TxRemoveOutput.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
LightningCodecs.writeU64(serialId, out)
}
companion object : LightningMessageReader {
const val type: Long = 69
override fun read(input: Input): TxRemoveOutput = TxRemoveOutput(
LightningCodecs.bytes(input, 32).byteVector32(),
LightningCodecs.u64(input),
)
}
}
data class TxComplete(
override val channelId: ByteVector32,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxConstructionMessage(), HasChannelId {
override val type: Long get() = TxComplete.type
val publicNonces: List = tlvs.get()?.nonces ?: listOf()
constructor(channelId: ByteVector32, publicNonces: List) : this(channelId, TlvStream(TxCompleteTlv.Nonces(publicNonces)))
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 70
@Suppress("UNCHECKED_CAST")
val readers = mapOf(TxCompleteTlv.Nonces.tag to TxCompleteTlv.Nonces.Companion as TlvValueReader)
override fun read(input: Input): TxComplete = TxComplete(LightningCodecs.bytes(input, 32).byteVector32(), TlvStreamSerializer(false, readers).read(input))
}
}
data class TxSignatures(
override val channelId: ByteVector32,
val txId: TxId,
val witnesses: List,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxMessage(), HasChannelId, HasEncryptedChannelData {
constructor(
channelId: ByteVector32,
tx: Transaction,
witnesses: List,
previousFundingSig: ByteVector64?,
swapInUserSigs: List,
swapInServerSigs: List,
swapInUserPartialSigs: List,
swapInServerPartialSigs: List
) : this(
channelId,
tx.txid,
witnesses,
TlvStream(
setOfNotNull(
previousFundingSig?.let { TxSignaturesTlv.PreviousFundingTxSig(it) },
if (swapInUserSigs.isNotEmpty()) TxSignaturesTlv.SwapInUserSigs(swapInUserSigs) else null,
if (swapInServerSigs.isNotEmpty()) TxSignaturesTlv.SwapInServerSigs(swapInServerSigs) else null,
if (swapInUserPartialSigs.isNotEmpty()) TxSignaturesTlv.SwapInUserPartialSigs(swapInUserPartialSigs) else null,
if (swapInServerPartialSigs.isNotEmpty()) TxSignaturesTlv.SwapInServerPartialSigs(swapInServerPartialSigs) else null,
)
),
)
override val type: Long get() = TxSignatures.type
val previousFundingTxSig: ByteVector64? = tlvs.get()?.sig
val swapInUserSigs: List = tlvs.get()?.sigs ?: listOf()
val swapInServerSigs: List = tlvs.get()?.sigs ?: listOf()
val swapInUserPartialSigs: List = tlvs.get()?.psigs ?: listOf()
val swapInServerPartialSigs: List = tlvs.get()?.psigs ?: listOf()
override val channelData: EncryptedChannelData get() = tlvs.get()?.ecb ?: EncryptedChannelData.empty
override fun withNonEmptyChannelData(ecd: EncryptedChannelData): TxSignatures = copy(tlvs = tlvs.addOrUpdate(TxSignaturesTlv.ChannelData(ecd)))
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
// Note that we encode the tx_hash instead of the tx_id to be consistent with other lightning messages.
LightningCodecs.writeTxHash(TxHash(txId), out)
LightningCodecs.writeU16(witnesses.size, out)
witnesses.forEach { witness ->
val witnessData = ScriptWitness.write(witness)
LightningCodecs.writeU16(witnessData.size, out)
LightningCodecs.writeBytes(witnessData, out)
}
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 71
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
TxSignaturesTlv.PreviousFundingTxSig.tag to TxSignaturesTlv.PreviousFundingTxSig.Companion as TlvValueReader,
TxSignaturesTlv.SwapInUserSigs.tag to TxSignaturesTlv.SwapInUserSigs.Companion as TlvValueReader,
TxSignaturesTlv.SwapInServerSigs.tag to TxSignaturesTlv.SwapInServerSigs.Companion as TlvValueReader,
TxSignaturesTlv.SwapInUserPartialSigs.tag to TxSignaturesTlv.SwapInUserPartialSigs.Companion as TlvValueReader,
TxSignaturesTlv.SwapInServerPartialSigs.tag to TxSignaturesTlv.SwapInServerPartialSigs.Companion as TlvValueReader,
TxSignaturesTlv.ChannelData.tag to TxSignaturesTlv.ChannelData.Companion as TlvValueReader,
)
override fun read(input: Input): TxSignatures {
val channelId = LightningCodecs.bytes(input, 32).byteVector32()
val txHash = LightningCodecs.txHash(input)
val witnessCount = LightningCodecs.u16(input)
val witnesses = (1..witnessCount).map {
val witnessSize = LightningCodecs.u16(input)
ScriptWitness.read(LightningCodecs.bytes(input, witnessSize))
}
return TxSignatures(channelId, TxId(txHash), witnesses, TlvStreamSerializer(false, readers).read(input))
}
}
}
data class TxInitRbf(
override val channelId: ByteVector32,
val lockTime: Long,
val feerate: FeeratePerKw,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxMessage(), HasChannelId {
constructor(channelId: ByteVector32, lockTime: Long, feerate: FeeratePerKw, fundingContribution: Satoshi) : this(channelId, lockTime, feerate, TlvStream(TxInitRbfTlv.SharedOutputContributionTlv(fundingContribution)))
val fundingContribution = tlvs.get()?.amount ?: 0.sat
override val type: Long get() = TxInitRbf.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
LightningCodecs.writeU32(lockTime.toInt(), out)
LightningCodecs.writeU32(feerate.toLong().toInt(), out)
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 72
@Suppress("UNCHECKED_CAST")
val readers = mapOf(TxInitRbfTlv.SharedOutputContributionTlv.tag to TxInitRbfTlv.SharedOutputContributionTlv.Companion as TlvValueReader)
override fun read(input: Input): TxInitRbf = TxInitRbf(
LightningCodecs.bytes(input, 32).byteVector32(),
LightningCodecs.u32(input).toLong(),
FeeratePerKw(LightningCodecs.u32(input).toLong().sat),
TlvStreamSerializer(false, readers).read(input),
)
}
}
data class TxAckRbf(
override val channelId: ByteVector32,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxMessage(), HasChannelId {
constructor(channelId: ByteVector32, fundingContribution: Satoshi) : this(channelId, TlvStream(TxAckRbfTlv.SharedOutputContributionTlv(fundingContribution)))
val fundingContribution = tlvs.get()?.amount ?: 0.sat
override val type: Long get() = TxAckRbf.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId.toByteArray(), out)
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 73
@Suppress("UNCHECKED_CAST")
val readers = mapOf(TxAckRbfTlv.SharedOutputContributionTlv.tag to TxAckRbfTlv.SharedOutputContributionTlv.Companion as TlvValueReader)
override fun read(input: Input): TxAckRbf = TxAckRbf(
LightningCodecs.bytes(input, 32).byteVector32(),
TlvStreamSerializer(false, readers).read(input),
)
}
}
data class TxAbort(
override val channelId: ByteVector32,
val data: ByteVector,
val tlvs: TlvStream = TlvStream.empty()
) : InteractiveTxMessage(), HasChannelId {
constructor(channelId: ByteVector32, message: String?) : this(channelId, ByteVector(message?.encodeToByteArray() ?: ByteArray(0)))
fun toAscii(): String = data.toByteArray().decodeToString()
override val type: Long get() = TxAbort.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU16(data.size(), out)
LightningCodecs.writeBytes(data, out)
}
companion object : LightningMessageReader {
const val type: Long = 74
override fun read(input: Input): TxAbort = TxAbort(
LightningCodecs.bytes(input, 32).byteVector32(),
LightningCodecs.bytes(input, LightningCodecs.u16(input)).byteVector(),
)
}
}
data class OpenDualFundedChannel(
override val chainHash: BlockHash,
override val temporaryChannelId: ByteVector32,
val fundingFeerate: FeeratePerKw,
val commitmentFeerate: FeeratePerKw,
val fundingAmount: Satoshi,
val dustLimit: Satoshi,
val maxHtlcValueInFlightMsat: Long, // this is not MilliSatoshi because it can exceed the total amount of MilliSatoshi
val htlcMinimum: MilliSatoshi,
val toSelfDelay: CltvExpiryDelta,
val maxAcceptedHtlcs: Int,
val lockTime: Long,
val fundingPubkey: PublicKey,
val revocationBasepoint: PublicKey,
val paymentBasepoint: PublicKey,
val delayedPaymentBasepoint: PublicKey,
val htlcBasepoint: PublicKey,
val firstPerCommitmentPoint: PublicKey,
val secondPerCommitmentPoint: PublicKey,
val channelFlags: Byte,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasTemporaryChannelId, HasChainHash {
val channelType: ChannelType? get() = tlvStream.get()?.channelType
val pushAmount: MilliSatoshi get() = tlvStream.get()?.amount ?: 0.msat
val requestFunds: ChannelTlv.RequestFunds? get() = tlvStream.get()
val origin: Origin? get() = tlvStream.get()?.origin
override val type: Long get() = OpenDualFundedChannel.type
override fun write(out: Output) {
LightningCodecs.writeBytes(chainHash.value, out)
LightningCodecs.writeBytes(temporaryChannelId, out)
LightningCodecs.writeU32(fundingFeerate.toLong().toInt(), out)
LightningCodecs.writeU32(commitmentFeerate.toLong().toInt(), out)
LightningCodecs.writeU64(fundingAmount.toLong(), out)
LightningCodecs.writeU64(dustLimit.toLong(), out)
LightningCodecs.writeU64(maxHtlcValueInFlightMsat, out)
LightningCodecs.writeU64(htlcMinimum.toLong(), out)
LightningCodecs.writeU16(toSelfDelay.toInt(), out)
LightningCodecs.writeU16(maxAcceptedHtlcs, out)
LightningCodecs.writeU32(lockTime.toInt(), out)
LightningCodecs.writeBytes(fundingPubkey.value, out)
LightningCodecs.writeBytes(revocationBasepoint.value, out)
LightningCodecs.writeBytes(paymentBasepoint.value, out)
LightningCodecs.writeBytes(delayedPaymentBasepoint.value, out)
LightningCodecs.writeBytes(htlcBasepoint.value, out)
LightningCodecs.writeBytes(firstPerCommitmentPoint.value, out)
LightningCodecs.writeBytes(secondPerCommitmentPoint.value, out)
LightningCodecs.writeByte(channelFlags.toInt(), out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 64
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
ChannelTlv.UpfrontShutdownScriptTlv.tag to ChannelTlv.UpfrontShutdownScriptTlv.Companion as TlvValueReader,
ChannelTlv.ChannelTypeTlv.tag to ChannelTlv.ChannelTypeTlv.Companion as TlvValueReader,
ChannelTlv.RequireConfirmedInputsTlv.tag to ChannelTlv.RequireConfirmedInputsTlv as TlvValueReader,
ChannelTlv.RequestFunds.tag to ChannelTlv.RequestFunds as TlvValueReader,
ChannelTlv.OriginTlv.tag to ChannelTlv.OriginTlv.Companion as TlvValueReader,
ChannelTlv.PushAmountTlv.tag to ChannelTlv.PushAmountTlv.Companion as TlvValueReader,
)
override fun read(input: Input): OpenDualFundedChannel = OpenDualFundedChannel(
BlockHash(LightningCodecs.bytes(input, 32)),
ByteVector32(LightningCodecs.bytes(input, 32)),
FeeratePerKw(LightningCodecs.u32(input).toLong().sat),
FeeratePerKw(LightningCodecs.u32(input).toLong().sat),
Satoshi(LightningCodecs.u64(input)),
Satoshi(LightningCodecs.u64(input)),
LightningCodecs.u64(input), // this is not MilliSatoshi because it can exceed the total amount of MilliSatoshi
MilliSatoshi(LightningCodecs.u64(input)),
CltvExpiryDelta(LightningCodecs.u16(input)),
LightningCodecs.u16(input),
LightningCodecs.u32(input).toLong(),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
LightningCodecs.byte(input).toByte(),
TlvStreamSerializer(false, readers).read(input)
)
}
}
data class AcceptDualFundedChannel(
override val temporaryChannelId: ByteVector32,
val fundingAmount: Satoshi,
val dustLimit: Satoshi,
val maxHtlcValueInFlightMsat: Long, // this is not MilliSatoshi because it can exceed the total amount of MilliSatoshi
val htlcMinimum: MilliSatoshi,
val minimumDepth: Long,
val toSelfDelay: CltvExpiryDelta,
val maxAcceptedHtlcs: Int,
val fundingPubkey: PublicKey,
val revocationBasepoint: PublicKey,
val paymentBasepoint: PublicKey,
val delayedPaymentBasepoint: PublicKey,
val htlcBasepoint: PublicKey,
val firstPerCommitmentPoint: PublicKey,
val secondPerCommitmentPoint: PublicKey,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasTemporaryChannelId {
val channelType: ChannelType? get() = tlvStream.get()?.channelType
val willFund: ChannelTlv.WillFund? get() = tlvStream.get()
val pushAmount: MilliSatoshi get() = tlvStream.get()?.amount ?: 0.msat
override val type: Long get() = AcceptDualFundedChannel.type
override fun write(out: Output) {
LightningCodecs.writeBytes(temporaryChannelId, out)
LightningCodecs.writeU64(fundingAmount.toLong(), out)
LightningCodecs.writeU64(dustLimit.toLong(), out)
LightningCodecs.writeU64(maxHtlcValueInFlightMsat, out)
LightningCodecs.writeU64(htlcMinimum.toLong(), out)
LightningCodecs.writeU32(minimumDepth.toInt(), out)
LightningCodecs.writeU16(toSelfDelay.toInt(), out)
LightningCodecs.writeU16(maxAcceptedHtlcs, out)
LightningCodecs.writeBytes(fundingPubkey.value, out)
LightningCodecs.writeBytes(revocationBasepoint.value, out)
LightningCodecs.writeBytes(paymentBasepoint.value, out)
LightningCodecs.writeBytes(delayedPaymentBasepoint.value, out)
LightningCodecs.writeBytes(htlcBasepoint.value, out)
LightningCodecs.writeBytes(firstPerCommitmentPoint.value, out)
LightningCodecs.writeBytes(secondPerCommitmentPoint.value, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 65
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
ChannelTlv.UpfrontShutdownScriptTlv.tag to ChannelTlv.UpfrontShutdownScriptTlv.Companion as TlvValueReader,
ChannelTlv.ChannelTypeTlv.tag to ChannelTlv.ChannelTypeTlv.Companion as TlvValueReader,
ChannelTlv.RequireConfirmedInputsTlv.tag to ChannelTlv.RequireConfirmedInputsTlv as TlvValueReader,
ChannelTlv.WillFund.tag to ChannelTlv.WillFund as TlvValueReader,
ChannelTlv.PushAmountTlv.tag to ChannelTlv.PushAmountTlv.Companion as TlvValueReader,
)
override fun read(input: Input): AcceptDualFundedChannel = AcceptDualFundedChannel(
ByteVector32(LightningCodecs.bytes(input, 32)),
Satoshi(LightningCodecs.u64(input)),
Satoshi(LightningCodecs.u64(input)),
LightningCodecs.u64(input), // this is not MilliSatoshi because it can exceed the total amount of MilliSatoshi,
MilliSatoshi(LightningCodecs.u64(input)),
LightningCodecs.u32(input).toLong(),
CltvExpiryDelta(LightningCodecs.u16(input)),
LightningCodecs.u16(input),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
PublicKey(LightningCodecs.bytes(input, 33)),
TlvStreamSerializer(false, readers).read(input)
)
}
}
data class FundingCreated(
override val temporaryChannelId: ByteVector32,
val fundingTxHash: ByteVector32,
val fundingOutputIndex: Int,
val signature: ByteVector64
) : ChannelMessage, HasTemporaryChannelId {
override val type: Long get() = FundingCreated.type
override fun write(out: Output) {
LightningCodecs.writeBytes(temporaryChannelId, out)
LightningCodecs.writeBytes(fundingTxHash, out)
LightningCodecs.writeU16(fundingOutputIndex, out)
LightningCodecs.writeBytes(signature, out)
}
companion object : LightningMessageReader {
const val type: Long = 34
override fun read(input: Input): FundingCreated {
return FundingCreated(
ByteVector32(LightningCodecs.bytes(input, 32)),
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.u16(input),
ByteVector64(LightningCodecs.bytes(input, 64))
)
}
}
}
data class FundingSigned(
override val channelId: ByteVector32,
val signature: ByteVector64,
) : ChannelMessage, HasChannelId {
override val type: Long get() = FundingSigned.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeBytes(signature, out)
}
companion object : LightningMessageReader {
const val type: Long = 35
override fun read(input: Input): FundingSigned {
return FundingSigned(
ByteVector32(LightningCodecs.bytes(input, 32)),
ByteVector64(LightningCodecs.bytes(input, 64)),
)
}
}
}
data class ChannelReady(
override val channelId: ByteVector32,
val nextPerCommitmentPoint: PublicKey,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasChannelId {
override val type: Long get() = ChannelReady.type
val alias: ShortChannelId? = tlvStream.get()?.alias
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeBytes(nextPerCommitmentPoint.value, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 36
@Suppress("UNCHECKED_CAST")
val readers = mapOf(ChannelReadyTlv.ShortChannelIdTlv.tag to ChannelReadyTlv.ShortChannelIdTlv.Companion as TlvValueReader)
override fun read(input: Input) = ChannelReady(
ByteVector32(LightningCodecs.bytes(input, 32)),
PublicKey(LightningCodecs.bytes(input, 33)),
TlvStreamSerializer(false, readers).read(input)
)
}
}
data class Stfu(
override val channelId: ByteVector32,
val initiator: Boolean
) : SetupMessage, HasChannelId {
override val type: Long get() = Stfu.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeByte(if (initiator) 1 else 0, out)
}
companion object : LightningMessageReader {
const val type: Long = 2
override fun read(input: Input): Stfu {
return Stfu(
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.byte(input) == 1
)
}
}
}
data class SpliceInit(
override val channelId: ByteVector32,
val fundingContribution: Satoshi,
val feerate: FeeratePerKw,
val lockTime: Long,
val fundingPubkey: PublicKey,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasChannelId {
override val type: Long get() = SpliceInit.type
val requireConfirmedInputs: Boolean = tlvStream.get()?.let { true } ?: false
val requestFunds: ChannelTlv.RequestFunds? get() = tlvStream.get()
val pushAmount: MilliSatoshi = tlvStream.get()?.amount ?: 0.msat
val origins: List = tlvStream.get()?.origins?.filterIsInstance() ?: emptyList()
constructor(channelId: ByteVector32, fundingContribution: Satoshi, pushAmount: MilliSatoshi, feerate: FeeratePerKw, lockTime: Long, fundingPubkey: PublicKey, requestFunds: ChannelTlv.RequestFunds?) : this(
channelId,
fundingContribution,
feerate,
lockTime,
fundingPubkey,
TlvStream(setOfNotNull(if (pushAmount > 0.msat) ChannelTlv.PushAmountTlv(pushAmount) else null, requestFunds))
)
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeInt64(fundingContribution.toLong(), out)
LightningCodecs.writeU32(feerate.toLong().toInt(), out)
LightningCodecs.writeU32(lockTime.toInt(), out)
LightningCodecs.writeBytes(fundingPubkey.value, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 37000
@Suppress("UNCHECKED_CAST")
private val readers = mapOf(
ChannelTlv.RequireConfirmedInputsTlv.tag to ChannelTlv.RequireConfirmedInputsTlv as TlvValueReader,
ChannelTlv.RequestFunds.tag to ChannelTlv.RequestFunds as TlvValueReader,
ChannelTlv.PushAmountTlv.tag to ChannelTlv.PushAmountTlv.Companion as TlvValueReader,
ChannelTlv.OriginsTlv.tag to ChannelTlv.OriginsTlv.Companion as TlvValueReader
)
override fun read(input: Input): SpliceInit = SpliceInit(
channelId = ByteVector32(LightningCodecs.bytes(input, 32)),
fundingContribution = Satoshi(LightningCodecs.int64(input)),
feerate = FeeratePerKw(LightningCodecs.u32(input).toLong().sat),
lockTime = LightningCodecs.u32(input).toLong(),
fundingPubkey = PublicKey(LightningCodecs.bytes(input, 33)),
tlvStream = TlvStreamSerializer(false, readers).read(input)
)
}
}
data class SpliceAck(
override val channelId: ByteVector32,
val fundingContribution: Satoshi,
val fundingPubkey: PublicKey,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasChannelId {
override val type: Long get() = SpliceAck.type
val requireConfirmedInputs: Boolean = tlvStream.get()?.let { true } ?: false
val willFund: ChannelTlv.WillFund? get() = tlvStream.get()
val pushAmount: MilliSatoshi = tlvStream.get()?.amount ?: 0.msat
constructor(channelId: ByteVector32, fundingContribution: Satoshi, pushAmount: MilliSatoshi, fundingPubkey: PublicKey, willFund: ChannelTlv.WillFund?) : this(
channelId,
fundingContribution,
fundingPubkey,
TlvStream(setOfNotNull(if (pushAmount > 0.msat) ChannelTlv.PushAmountTlv(pushAmount) else null, willFund))
)
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeInt64(fundingContribution.toLong(), out)
LightningCodecs.writeBytes(fundingPubkey.value, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 37002
@Suppress("UNCHECKED_CAST")
private val readers = mapOf(
ChannelTlv.RequireConfirmedInputsTlv.tag to ChannelTlv.RequireConfirmedInputsTlv as TlvValueReader,
ChannelTlv.WillFund.tag to ChannelTlv.WillFund as TlvValueReader,
ChannelTlv.PushAmountTlv.tag to ChannelTlv.PushAmountTlv.Companion as TlvValueReader,
)
override fun read(input: Input): SpliceAck = SpliceAck(
channelId = ByteVector32(LightningCodecs.bytes(input, 32)),
fundingContribution = Satoshi(LightningCodecs.int64(input)),
fundingPubkey = PublicKey(LightningCodecs.bytes(input, 33)),
tlvStream = TlvStreamSerializer(false, readers).read(input)
)
}
}
data class SpliceLocked(
override val channelId: ByteVector32,
val fundingTxId: TxId,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasChannelId {
override val type: Long get() = SpliceLocked.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeTxHash(TxHash(fundingTxId), out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 37004
private val readers = emptyMap>()
override fun read(input: Input): SpliceLocked = SpliceLocked(
channelId = ByteVector32(LightningCodecs.bytes(input, 32)),
fundingTxId = TxId(LightningCodecs.txHash(input)),
tlvStream = TlvStreamSerializer(false, readers).read(input)
)
}
}
data class UpdateAddHtlc(
override val channelId: ByteVector32,
val id: Long,
val amountMsat: MilliSatoshi,
val paymentHash: ByteVector32,
val cltvExpiry: CltvExpiry,
val onionRoutingPacket: OnionRoutingPacket
) : HtlcMessage, UpdateMessage, HasChannelId, ForbiddenMessageDuringSplice {
override val type: Long get() = UpdateAddHtlc.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(id, out)
LightningCodecs.writeU64(amountMsat.toLong(), out)
LightningCodecs.writeBytes(paymentHash, out)
LightningCodecs.writeU32(cltvExpiry.toLong().toInt(), out)
OnionRoutingPacketSerializer(OnionRoutingPacket.PaymentPacketLength).write(onionRoutingPacket, out)
}
companion object : LightningMessageReader {
const val type: Long = 128
override fun read(input: Input): UpdateAddHtlc {
val channelId = ByteVector32(LightningCodecs.bytes(input, 32))
val id = LightningCodecs.u64(input)
val amount = MilliSatoshi(LightningCodecs.u64(input))
val paymentHash = ByteVector32(LightningCodecs.bytes(input, 32))
val expiry = CltvExpiry(LightningCodecs.u32(input).toLong())
val onion = OnionRoutingPacketSerializer(OnionRoutingPacket.PaymentPacketLength).read(input)
return UpdateAddHtlc(channelId, id, amount, paymentHash, expiry, onion)
}
}
}
data class UpdateFulfillHtlc(
override val channelId: ByteVector32,
override val id: Long,
val paymentPreimage: ByteVector32
) : HtlcMessage, HtlcSettlementMessage, HasChannelId {
override val type: Long get() = UpdateFulfillHtlc.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(id, out)
LightningCodecs.writeBytes(paymentPreimage, out)
}
companion object : LightningMessageReader {
const val type: Long = 130
override fun read(input: Input): UpdateFulfillHtlc {
return UpdateFulfillHtlc(
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.u64(input),
ByteVector32(LightningCodecs.bytes(input, 32))
)
}
}
}
data class UpdateFailHtlc(
override val channelId: ByteVector32,
override val id: Long,
val reason: ByteVector
) : HtlcMessage, HtlcSettlementMessage, HasChannelId {
override val type: Long get() = UpdateFailHtlc.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(id, out)
LightningCodecs.writeU16(reason.size(), out)
LightningCodecs.writeBytes(reason, out)
}
companion object : LightningMessageReader {
const val type: Long = 131
override fun read(input: Input): UpdateFailHtlc {
return UpdateFailHtlc(
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.u64(input),
ByteVector(LightningCodecs.bytes(input, LightningCodecs.u16(input)))
)
}
}
}
data class UpdateFailMalformedHtlc(
override val channelId: ByteVector32,
override val id: Long,
val onionHash: ByteVector32,
val failureCode: Int
) : HtlcMessage, HtlcSettlementMessage, HasChannelId {
override val type: Long get() = UpdateFailMalformedHtlc.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(id, out)
LightningCodecs.writeBytes(onionHash, out)
LightningCodecs.writeU16(failureCode, out)
}
companion object : LightningMessageReader {
const val type: Long = 135
override fun read(input: Input): UpdateFailMalformedHtlc {
return UpdateFailMalformedHtlc(
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.u64(input),
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.u16(input)
)
}
}
}
data class CommitSig(
override val channelId: ByteVector32,
val signature: ByteVector64,
val htlcSignatures: List,
val tlvStream: TlvStream = TlvStream.empty()
) : HtlcMessage, HasChannelId, HasEncryptedChannelData {
override val type: Long get() = CommitSig.type
override val channelData: EncryptedChannelData get() = tlvStream.get()?.ecb ?: EncryptedChannelData.empty
override fun withNonEmptyChannelData(ecd: EncryptedChannelData): CommitSig = copy(tlvStream = tlvStream.addOrUpdate(CommitSigTlv.ChannelData(ecd)))
val alternativeFeerateSigs: List = tlvStream.get()?.sigs ?: listOf()
val batchSize: Int = tlvStream.get()?.size ?: 1
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeBytes(signature, out)
LightningCodecs.writeU16(htlcSignatures.size, out)
htlcSignatures.forEach { LightningCodecs.writeBytes(it, out) }
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 132
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
CommitSigTlv.ChannelData.tag to CommitSigTlv.ChannelData.Companion as TlvValueReader,
CommitSigTlv.AlternativeFeerateSigs.tag to CommitSigTlv.AlternativeFeerateSigs.Companion as TlvValueReader,
CommitSigTlv.Batch.tag to CommitSigTlv.Batch.Companion as TlvValueReader,
)
override fun read(input: Input): CommitSig {
val channelId = ByteVector32(LightningCodecs.bytes(input, 32))
val sig = ByteVector64(LightningCodecs.bytes(input, 64))
val numHtlcs = LightningCodecs.u16(input)
val htlcSigs = ArrayList(numHtlcs)
for (i in 1..numHtlcs) {
htlcSigs += ByteVector64(LightningCodecs.bytes(input, 64))
}
return CommitSig(channelId, sig, htlcSigs.toList(), TlvStreamSerializer(false, readers).read(input))
}
}
}
data class RevokeAndAck(
override val channelId: ByteVector32,
val perCommitmentSecret: PrivateKey,
val nextPerCommitmentPoint: PublicKey,
val tlvStream: TlvStream = TlvStream.empty()
) : HtlcMessage, HasChannelId, HasEncryptedChannelData {
override val type: Long get() = RevokeAndAck.type
override val channelData: EncryptedChannelData get() = tlvStream.get()?.ecb ?: EncryptedChannelData.empty
override fun withNonEmptyChannelData(ecd: EncryptedChannelData): RevokeAndAck = copy(tlvStream = tlvStream.addOrUpdate(RevokeAndAckTlv.ChannelData(ecd)))
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeBytes(perCommitmentSecret.value, out)
LightningCodecs.writeBytes(nextPerCommitmentPoint.value, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 133
@Suppress("UNCHECKED_CAST")
val readers = mapOf(RevokeAndAckTlv.ChannelData.tag to RevokeAndAckTlv.ChannelData.Companion as TlvValueReader)
override fun read(input: Input): RevokeAndAck {
return RevokeAndAck(
ByteVector32(LightningCodecs.bytes(input, 32)),
PrivateKey(LightningCodecs.bytes(input, 32)),
PublicKey(LightningCodecs.bytes(input, 33)),
TlvStreamSerializer(false, readers).read(input)
)
}
}
}
data class UpdateFee(
override val channelId: ByteVector32,
val feeratePerKw: FeeratePerKw
) : ChannelMessage, UpdateMessage, HasChannelId {
override val type: Long get() = UpdateFee.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU32(feeratePerKw.feerate.toLong().toInt(), out)
}
companion object : LightningMessageReader {
const val type: Long = 134
override fun read(input: Input): UpdateFee {
return UpdateFee(
ByteVector32(LightningCodecs.bytes(input, 32)),
FeeratePerKw(LightningCodecs.u32(input).toLong().sat)
)
}
}
}
data class ChannelReestablish(
override val channelId: ByteVector32,
val nextLocalCommitmentNumber: Long,
val nextRemoteRevocationNumber: Long,
val yourLastCommitmentSecret: PrivateKey,
val myCurrentPerCommitmentPoint: PublicKey,
val tlvStream: TlvStream = TlvStream.empty()
) : HasChannelId, HasEncryptedChannelData {
override val type: Long get() = ChannelReestablish.type
val nextFundingTxId: TxId? = tlvStream.get()?.txId
override val channelData: EncryptedChannelData get() = tlvStream.get()?.ecb ?: EncryptedChannelData.empty
override fun withNonEmptyChannelData(ecd: EncryptedChannelData): ChannelReestablish = copy(tlvStream = tlvStream.addOrUpdate(ChannelReestablishTlv.ChannelData(ecd)))
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(nextLocalCommitmentNumber, out)
LightningCodecs.writeU64(nextRemoteRevocationNumber, out)
LightningCodecs.writeBytes(yourLastCommitmentSecret.value, out)
LightningCodecs.writeBytes(myCurrentPerCommitmentPoint.value, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 136
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
ChannelReestablishTlv.ChannelData.tag to ChannelReestablishTlv.ChannelData.Companion as TlvValueReader,
ChannelReestablishTlv.NextFunding.tag to ChannelReestablishTlv.NextFunding.Companion as TlvValueReader,
)
override fun read(input: Input): ChannelReestablish {
return ChannelReestablish(
ByteVector32(LightningCodecs.bytes(input, 32)),
LightningCodecs.u64(input),
LightningCodecs.u64(input),
PrivateKey(LightningCodecs.bytes(input, 32)),
PublicKey(LightningCodecs.bytes(input, 33)),
TlvStreamSerializer(false, readers).read(input)
)
}
}
}
data class AnnouncementSignatures(
override val channelId: ByteVector32,
val shortChannelId: ShortChannelId,
val nodeSignature: ByteVector64,
val bitcoinSignature: ByteVector64
) : RoutingMessage, HasChannelId {
override val type: Long get() = AnnouncementSignatures.type
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(shortChannelId.toLong(), out)
LightningCodecs.writeBytes(nodeSignature, out)
LightningCodecs.writeBytes(bitcoinSignature, out)
}
companion object : LightningMessageReader {
const val type: Long = 259
override fun read(input: Input): AnnouncementSignatures {
return AnnouncementSignatures(
ByteVector32(LightningCodecs.bytes(input, 32)),
ShortChannelId(LightningCodecs.u64(input)),
ByteVector64(LightningCodecs.bytes(input, 64)),
ByteVector64(LightningCodecs.bytes(input, 64))
)
}
}
}
data class ChannelAnnouncement(
val nodeSignature1: ByteVector64,
val nodeSignature2: ByteVector64,
val bitcoinSignature1: ByteVector64,
val bitcoinSignature2: ByteVector64,
val features: Features,
override val chainHash: BlockHash,
val shortChannelId: ShortChannelId,
val nodeId1: PublicKey,
val nodeId2: PublicKey,
val bitcoinKey1: PublicKey,
val bitcoinKey2: PublicKey,
val unknownFields: ByteVector = ByteVector.empty
) : AnnouncementMessage, HasChainHash {
override val type: Long get() = ChannelAnnouncement.type
override fun write(out: Output) {
LightningCodecs.writeBytes(nodeSignature1, out)
LightningCodecs.writeBytes(nodeSignature2, out)
LightningCodecs.writeBytes(bitcoinSignature1, out)
LightningCodecs.writeBytes(bitcoinSignature2, out)
val featureBytes = features.toByteArray()
LightningCodecs.writeU16(featureBytes.size, out)
LightningCodecs.writeBytes(featureBytes, out)
LightningCodecs.writeBytes(chainHash.value, out)
LightningCodecs.writeU64(shortChannelId.toLong(), out)
LightningCodecs.writeBytes(nodeId1.value, out)
LightningCodecs.writeBytes(nodeId2.value, out)
LightningCodecs.writeBytes(bitcoinKey1.value, out)
LightningCodecs.writeBytes(bitcoinKey2.value, out)
LightningCodecs.writeBytes(unknownFields, out)
}
companion object : LightningMessageReader {
const val type: Long = 256
override fun read(input: Input): ChannelAnnouncement {
val nodeSignature1 = LightningCodecs.bytes(input, 64).toByteVector64()
val nodeSignature2 = LightningCodecs.bytes(input, 64).toByteVector64()
val bitcoinSignature1 = LightningCodecs.bytes(input, 64).toByteVector64()
val bitcoinSignature2 = LightningCodecs.bytes(input, 64).toByteVector64()
val featureBytes = LightningCodecs.bytes(input, LightningCodecs.u16(input))
val chainHash = BlockHash(LightningCodecs.bytes(input, 32))
val shortChannelId = ShortChannelId(LightningCodecs.u64(input))
val nodeId1 = PublicKey(LightningCodecs.bytes(input, 33))
val nodeId2 = PublicKey(LightningCodecs.bytes(input, 33))
val bitcoinKey1 = PublicKey(LightningCodecs.bytes(input, 33))
val bitcoinKey2 = PublicKey(LightningCodecs.bytes(input, 33))
val unknownBytes = if (input.availableBytes > 0) LightningCodecs.bytes(input, input.availableBytes).toByteVector() else ByteVector.empty
return ChannelAnnouncement(
nodeSignature1,
nodeSignature2,
bitcoinSignature1,
bitcoinSignature2,
Features(featureBytes),
chainHash,
shortChannelId,
nodeId1,
nodeId2,
bitcoinKey1,
bitcoinKey2,
unknownBytes
)
}
}
}
data class ChannelUpdate(
val signature: ByteVector64,
override val chainHash: BlockHash,
val shortChannelId: ShortChannelId,
override val timestampSeconds: Long,
val messageFlags: Byte,
val channelFlags: Byte,
val cltvExpiryDelta: CltvExpiryDelta,
val htlcMinimumMsat: MilliSatoshi,
val feeBaseMsat: MilliSatoshi,
val feeProportionalMillionths: Long,
val htlcMaximumMsat: MilliSatoshi?,
val unknownFields: ByteVector = ByteVector.empty
) : AnnouncementMessage, HasTimestamp, HasChainHash {
init {
require(((messageFlags.toInt() and 1) != 0) == (htlcMaximumMsat != null)) { "htlcMaximumMsat is not consistent with messageFlags" }
}
override val type: Long get() = ChannelUpdate.type
/** BOLT 7: The creating node [...] MUST set the direction bit of flags to 0 if the creating node is node-id-1 in that message, otherwise 1. */
fun isNode1(): Boolean = (channelFlags.toInt() and 1) == 0
/** BOLT 7: A node MAY create and send a channel_update with the disable bit set to signal the temporary unavailability of a channel */
fun isEnabled(): Boolean = (channelFlags.toInt() and 2) == 0
fun isRemote(localNodeId: PublicKey, remoteNodeId: PublicKey): Boolean = isNode1() != Announcements.isNode1(localNodeId, remoteNodeId)
override fun write(out: Output) {
LightningCodecs.writeBytes(signature, out)
LightningCodecs.writeBytes(chainHash.value, out)
LightningCodecs.writeU64(shortChannelId.toLong(), out)
LightningCodecs.writeU32(timestampSeconds.toInt(), out)
LightningCodecs.writeByte(messageFlags.toInt(), out)
LightningCodecs.writeByte(channelFlags.toInt(), out)
LightningCodecs.writeU16(cltvExpiryDelta.toInt(), out)
LightningCodecs.writeU64(htlcMinimumMsat.toLong(), out)
LightningCodecs.writeU32(feeBaseMsat.toLong().toInt(), out)
LightningCodecs.writeU32(feeProportionalMillionths.toInt(), out)
if (htlcMaximumMsat != null) {
LightningCodecs.writeU64(htlcMaximumMsat.toLong(), out)
}
LightningCodecs.writeBytes(unknownFields, out)
}
companion object : LightningMessageReader {
const val type: Long = 258
override fun read(input: Input): ChannelUpdate {
val signature = ByteVector64(LightningCodecs.bytes(input, 64))
val chainHash = BlockHash(LightningCodecs.bytes(input, 32))
val shortChannelId = ShortChannelId(LightningCodecs.u64(input))
val timestampSeconds = LightningCodecs.u32(input).toLong()
val messageFlags = LightningCodecs.byte(input).toByte()
val channelFlags = LightningCodecs.byte(input).toByte()
val cltvExpiryDelta = CltvExpiryDelta(LightningCodecs.u16(input))
val htlcMinimumMsat = MilliSatoshi(LightningCodecs.u64(input))
val feeBaseMsat = MilliSatoshi(LightningCodecs.u32(input).toLong())
val feeProportionalMillionths = LightningCodecs.u32(input).toLong()
val htlcMaximumMsat = if ((messageFlags.toInt() and 1) != 0) MilliSatoshi(LightningCodecs.u64(input)) else null
val unknownBytes = if (input.availableBytes > 0) LightningCodecs.bytes(input, input.availableBytes).toByteVector() else ByteVector.empty
return ChannelUpdate(
signature,
chainHash,
shortChannelId,
timestampSeconds,
messageFlags,
channelFlags,
cltvExpiryDelta,
htlcMinimumMsat,
feeBaseMsat,
feeProportionalMillionths,
htlcMaximumMsat,
unknownBytes
)
}
}
}
data class Shutdown(
override val channelId: ByteVector32,
val scriptPubKey: ByteVector,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasChannelId, HasEncryptedChannelData, ForbiddenMessageDuringSplice {
override val type: Long get() = Shutdown.type
override val channelData: EncryptedChannelData get() = tlvStream.get()?.ecb ?: EncryptedChannelData.empty
override fun withNonEmptyChannelData(ecd: EncryptedChannelData): Shutdown = copy(tlvStream = tlvStream.addOrUpdate(ShutdownTlv.ChannelData(ecd)))
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU16(scriptPubKey.size(), out)
LightningCodecs.writeBytes(scriptPubKey, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 38
@Suppress("UNCHECKED_CAST")
val readers = mapOf(ShutdownTlv.ChannelData.tag to ShutdownTlv.ChannelData.Companion as TlvValueReader)
override fun read(input: Input): Shutdown {
return Shutdown(
ByteVector32(LightningCodecs.bytes(input, 32)),
ByteVector(LightningCodecs.bytes(input, LightningCodecs.u16(input))),
TlvStreamSerializer(false, readers).read(input)
)
}
}
}
data class ClosingSigned(
override val channelId: ByteVector32,
val feeSatoshis: Satoshi,
val signature: ByteVector64,
val tlvStream: TlvStream = TlvStream.empty()
) : ChannelMessage, HasChannelId, HasEncryptedChannelData {
override val type: Long get() = ClosingSigned.type
override val channelData: EncryptedChannelData get() = tlvStream.get()?.ecb ?: EncryptedChannelData.empty
override fun withNonEmptyChannelData(ecd: EncryptedChannelData): ClosingSigned = copy(tlvStream = tlvStream.addOrUpdate(ClosingSignedTlv.ChannelData(ecd)))
override fun write(out: Output) {
LightningCodecs.writeBytes(channelId, out)
LightningCodecs.writeU64(feeSatoshis.toLong(), out)
LightningCodecs.writeBytes(signature, out)
TlvStreamSerializer(false, readers).write(tlvStream, out)
}
companion object : LightningMessageReader {
const val type: Long = 39
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
ClosingSignedTlv.FeeRange.tag to ClosingSignedTlv.FeeRange.Companion as TlvValueReader,
ClosingSignedTlv.ChannelData.tag to ClosingSignedTlv.ChannelData.Companion as TlvValueReader
)
override fun read(input: Input): ClosingSigned {
return ClosingSigned(
ByteVector32(LightningCodecs.bytes(input, 32)),
Satoshi(LightningCodecs.u64(input)),
ByteVector64(LightningCodecs.bytes(input, 64)),
TlvStreamSerializer(false, readers).read(input)
)
}
}
}
data class OnionMessage(
val blindingKey: PublicKey,
val onionRoutingPacket: OnionRoutingPacket
) : LightningMessage {
override val type: Long get() = OnionMessage.type
override fun write(out: Output) {
LightningCodecs.writeBytes(blindingKey.value, out)
LightningCodecs.writeU16(onionRoutingPacket.payload.size(), out)
OnionRoutingPacketSerializer(onionRoutingPacket.payload.size()).write(onionRoutingPacket, out)
}
companion object : LightningMessageReader {
const val type: Long = 513
override fun read(input: Input): OnionMessage {
val blindingKey = PublicKey(LightningCodecs.bytes(input, 33))
val onion = OnionRoutingPacketSerializer(LightningCodecs.u16(input) - 66).read(input)
return OnionMessage(blindingKey, onion)
}
}
}
/**
* When we don't have enough incoming liquidity to receive a payment, our peer may open a channel to us on-the-fly to carry that payment.
* This message contains details that allow us to recalculate the fee that our peer will take in exchange for the new channel.
* This allows us to combine multiple requests for the same payment and figure out the final fee that will be applied.
*
* @param chainHash chain we're on.
* @param fundingSatoshis total capacity of the channel our peer will open to us (some of the funds may be on their side).
* @param amountMsat payment amount covered by this new channel: we will receive push_msat = amountMsat - fees.
* @param payToOpenMinAmountMsat minimum amount for a pay-to-open to be attempted, this should be compared to the total amount in the case of an MPP payment.
* @param payToOpenFeeSatoshis fees that will be deducted from the amount pushed to us (this fee covers the on-chain fees our peer will pay to open the channel).
* @param paymentHash payment hash.
* @param expireAt after the proposal expires, our peer will fail the payment and won't open a channel to us.
* @param finalPacket onion packet that we would have received if there had been a channel to forward the payment to.
*/
data class PayToOpenRequest(
override val chainHash: BlockHash,
val amountMsat: MilliSatoshi,
val payToOpenFeeSatoshis: Satoshi,
val paymentHash: ByteVector32,
val expireAt: Long,
val finalPacket: OnionRoutingPacket
) : LightningMessage, HasChainHash {
override val type: Long get() = PayToOpenRequest.type
override fun write(out: Output) {
LightningCodecs.writeBytes(chainHash.value, out)
LightningCodecs.writeU64(0, out) // backward compat for removed field fundingSatoshis
LightningCodecs.writeU64(amountMsat.toLong(), out)
LightningCodecs.writeU64(0, out) // backward compat for removed field payToOpenMinAmountMsat
LightningCodecs.writeU64(payToOpenFeeSatoshis.toLong(), out)
LightningCodecs.writeBytes(paymentHash, out)
LightningCodecs.writeU32(expireAt.toInt(), out)
LightningCodecs.writeU16(finalPacket.payload.size(), out)
OnionRoutingPacketSerializer(finalPacket.payload.size()).write(finalPacket, out)
}
companion object : LightningMessageReader {
const val type: Long = 35021
override fun read(input: Input): PayToOpenRequest {
return PayToOpenRequest(
chainHash = BlockHash(LightningCodecs.bytes(input, 32))
.also { LightningCodecs.u64(input) }, // ignoring removed field fundingSatoshis
amountMsat = MilliSatoshi(LightningCodecs.u64(input))
.also { LightningCodecs.u64(input) }, // ignoring removed field payToOpenMinAmountMsat
payToOpenFeeSatoshis = Satoshi(LightningCodecs.u64(input)),
paymentHash = ByteVector32(LightningCodecs.bytes(input, 32)),
expireAt = LightningCodecs.u32(input).toLong(),
finalPacket = OnionRoutingPacketSerializer(LightningCodecs.u16(input)).read(input)
)
}
}
}
data class PayToOpenResponse(override val chainHash: BlockHash, val paymentHash: ByteVector32, val result: Result) : LightningMessage, HasChainHash {
override val type: Long get() = PayToOpenResponse.type
sealed class Result {
// @formatter:off
data class Success(val paymentPreimage: ByteVector32, val addToFeeCredit: Boolean = false) : Result()
/** reason is an onion-encrypted failure message, like those in UpdateFailHtlc */
data class Failure(val reason: ByteVector?) : Result()
// @formatter:on
}
override fun write(out: Output) {
LightningCodecs.writeBytes(chainHash.value, out)
LightningCodecs.writeBytes(paymentHash, out)
when (result) {
is Result.Success -> {
LightningCodecs.writeBytes(result.paymentPreimage, out)
LightningCodecs.writeByte(if (result.addToFeeCredit) 0xff else 0, out)
}
is Result.Failure -> {
LightningCodecs.writeBytes(ByteVector32.Zeroes, out) // this is for backward compatibility
result.reason?.let {
LightningCodecs.writeU16(it.size(), out)
LightningCodecs.writeBytes(it, out)
}
}
}
}
companion object : LightningMessageReader {
const val type: Long = 35003
override fun read(input: Input): PayToOpenResponse {
val chainHash = BlockHash(LightningCodecs.bytes(input, 32))
val paymentHash = LightningCodecs.bytes(input, 32).toByteVector32()
return when (val preimage = LightningCodecs.bytes(input, 32).toByteVector32()) {
ByteVector32.Zeroes -> {
val failure = if (input.availableBytes > 0) LightningCodecs.bytes(input, LightningCodecs.u16(input)).toByteVector() else null
PayToOpenResponse(chainHash, paymentHash, Result.Failure(failure))
}
else -> {
val addToFeeCredit = LightningCodecs.byte(input) != 0
PayToOpenResponse(chainHash, paymentHash, Result.Success(preimage, addToFeeCredit))
}
}
}
}
}
data class FCMToken(val token: ByteVector) : LightningMessage {
constructor(token: String) : this(ByteVector(token.encodeToByteArray()))
override val type: Long get() = FCMToken.type
override fun write(out: Output) {
LightningCodecs.writeU16(token.size(), out)
LightningCodecs.writeBytes(token, out)
}
fun toAscii(): String = token.toByteArray().decodeToString()
companion object : LightningMessageReader {
const val type: Long = 35017
override fun read(input: Input): FCMToken {
return FCMToken(LightningCodecs.bytes(input, LightningCodecs.u16(input)).toByteVector())
}
}
}
object UnsetFCMToken : LightningMessage {
override val type: Long get() = 35019
override fun write(out: Output) {}
}
data class PhoenixAndroidLegacyInfo(
val hasChannels: Boolean
) : LightningMessage {
override val type: Long get() = PhoenixAndroidLegacyInfo.type
override fun write(out: Output) {
LightningCodecs.writeByte(if (hasChannels) 0xff else 0, out)
}
companion object : LightningMessageReader {
const val type: Long = 35023
override fun read(input: Input): PhoenixAndroidLegacyInfo {
return PhoenixAndroidLegacyInfo(LightningCodecs.byte(input) != 0)
}
}
}
/**
* This message is used to request a channel open from a remote node, with local contributions to the funding transaction.
* If the remote node won't open a channel, it will respond with [PleaseOpenChannelRejected].
* Otherwise, it will respond with [OpenDualFundedChannel] and a fee that must be paid by a corresponding push_amount
* in the [AcceptDualFundedChannel] message.
*/
data class PleaseOpenChannel(
override val chainHash: BlockHash,
val requestId: ByteVector32,
val localFundingAmount: Satoshi,
val localInputsCount: Int,
val localInputsWeight: Int,
val tlvs: TlvStream = TlvStream.empty(),
) : LightningMessage, HasChainHash {
override val type: Long get() = PleaseOpenChannel.type
val grandParents: List = tlvs.get()?.outpoints ?: listOf()
override fun write(out: Output) {
LightningCodecs.writeBytes(chainHash.value, out)
LightningCodecs.writeBytes(requestId.toByteArray(), out)
LightningCodecs.writeU64(localFundingAmount.toLong(), out)
LightningCodecs.writeU16(localInputsCount, out)
LightningCodecs.writeU32(localInputsWeight, out)
TlvStreamSerializer(false, readers).write(tlvs, out)
}
companion object : LightningMessageReader {
const val type: Long = 36001
@Suppress("UNCHECKED_CAST")
val readers = mapOf(
PleaseOpenChannelTlv.GrandParents.tag to PleaseOpenChannelTlv.GrandParents.Companion as TlvValueReader,
)
override fun read(input: Input): PleaseOpenChannel = PleaseOpenChannel(
BlockHash(LightningCodecs.bytes(input, 32)),
LightningCodecs.bytes(input, 32).toByteVector32(),
LightningCodecs.u64(input).sat,
LightningCodecs.u16(input),
LightningCodecs.u32(input),
TlvStreamSerializer(false, readers).read(input)
)
}
}
data class CurrentFeeCredit(val amount: Satoshi) : LightningMessage {
override val type: Long get() = FCMToken.type
override fun write(out: Output) {
LightningCodecs.writeU64(amount.toLong(), out)
}
companion object : LightningMessageReader {
const val type: Long = 36003
override fun read(input: Input): CurrentFeeCredit {
return CurrentFeeCredit(LightningCodecs.u64(input).sat)
}
}
}
data class AutoLiquidityParams(val amount: Satoshi) : LightningMessage {
override val type: Long get() = AutoLiquidityParams.type
override fun write(out: Output) {
LightningCodecs.writeU64(amount.toLong(), out)
}
companion object : LightningMessageReader {
const val type: Long = 36005
override fun read(input: Input): AutoLiquidityParams {
return AutoLiquidityParams(LightningCodecs.u64(input).sat)
}
}
}
data class UnknownMessage(override val type: Long) : LightningMessage {
override fun write(out: Output) = TODO("Serialization of unknown messages is not implemented")
}