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

commonMain.fr.acinq.lightning.message.OnionMessages.kt Maven / Gradle / Ivy

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

import fr.acinq.bitcoin.*
import fr.acinq.bitcoin.utils.Either
import fr.acinq.lightning.EncodedNodeId
import fr.acinq.lightning.ShortChannelId
import fr.acinq.lightning.crypto.RouteBlinding
import fr.acinq.lightning.crypto.sphinx.Sphinx
import fr.acinq.lightning.logging.MDCLogger
import fr.acinq.lightning.utils.toByteVector
import fr.acinq.lightning.wire.*

object OnionMessages {
    data class IntermediateNode(val nodeId: EncodedNodeId.WithPublicKey, val outgoingChannelId: ShortChannelId? = null, val padding: ByteVector? = null, val customTlvs: Set = setOf()) {
        fun toTlvStream(nextNodeId: EncodedNodeId, nextBlinding: PublicKey? = null): TlvStream {
            val tlvs = setOfNotNull(
                outgoingChannelId?.let { RouteBlindingEncryptedDataTlv.OutgoingChannelId(it) } ?: RouteBlindingEncryptedDataTlv.OutgoingNodeId(nextNodeId),
                nextBlinding?.let { RouteBlindingEncryptedDataTlv.NextBlinding(it) },
                padding?.let { RouteBlindingEncryptedDataTlv.Padding(it) },
            )
            return TlvStream(tlvs, customTlvs)
        }
    }

    sealed class Destination {
        data class BlindedPath(val route: RouteBlinding.BlindedRoute) : Destination()
        data class Recipient(val nodeId: EncodedNodeId.WithPublicKey, val pathId: ByteVector?, val padding: ByteVector? = null, val customTlvs: Set = setOf()) : Destination()

        companion object {
            operator fun invoke(contactInfo: OfferTypes.ContactInfo): Destination =
                when (contactInfo) {
                    is OfferTypes.ContactInfo.BlindedPath -> BlindedPath(contactInfo.route)
                    is OfferTypes.ContactInfo.RecipientNodeId -> Recipient(EncodedNodeId.WithPublicKey.Plain(contactInfo.nodeId), null)
                }
        }
    }

    private fun buildIntermediatePayloads(
        intermediateNodes: List,
        lastNodeId: EncodedNodeId,
        lastBlinding: PublicKey? = null
    ): List {
        return if (intermediateNodes.isEmpty()) {
            listOf()
        } else {
            val intermediatePayloads = intermediateNodes.dropLast(1).zip(intermediateNodes.drop(1)).map { (current, next) ->
                current.toTlvStream(next.nodeId)
            }
            // The last intermediate node may contain a blinding override when the recipient is hidden behind a blinded path.
            val lastPayload = intermediateNodes.last().toTlvStream(lastNodeId, lastBlinding)
            (intermediatePayloads + lastPayload).map { RouteBlindingEncryptedData(it).write().byteVector() }
        }
    }

    fun buildRouteToRecipient(
        blindingSecret: PrivateKey,
        intermediateNodes: List,
        recipient: Destination.Recipient
    ): RouteBlinding. BlindedRouteDetails {
        val intermediatePayloads = buildIntermediatePayloads(intermediateNodes, recipient.nodeId)
        val tlvs = setOfNotNull(
            recipient.padding?.let { RouteBlindingEncryptedDataTlv.Padding(it) },
            recipient.pathId?.let { RouteBlindingEncryptedDataTlv.PathId(it) }
        )
        val lastPayload = RouteBlindingEncryptedData(TlvStream(tlvs, recipient.customTlvs)).write().toByteVector()
        return RouteBlinding.create(
            blindingSecret,
            intermediateNodes.map { it.nodeId.publicKey } + recipient.nodeId.publicKey,
            intermediatePayloads + lastPayload
        )
    }

    fun buildRoute(
        blindingSecret: PrivateKey,
        intermediateNodes: List,
        destination: Destination
    ): RouteBlinding.BlindedRoute {
        return when (destination) {
            is Destination.Recipient -> {
                buildRouteToRecipient(blindingSecret, intermediateNodes, destination).route
            }
            is Destination.BlindedPath -> when {
                intermediateNodes.isEmpty() -> destination.route
                else -> {
                    // We concatenate our blinded path with the destination's blinded path.
                    val intermediatePayloads = buildIntermediatePayloads(
                        intermediateNodes,
                        destination.route.introductionNodeId,
                        destination.route.blindingKey
                    )
                    val routePrefix = RouteBlinding.create(
                        blindingSecret,
                        intermediateNodes.map { it.nodeId.publicKey },
                        intermediatePayloads
                    ).route
                    RouteBlinding.BlindedRoute(
                        routePrefix.introductionNodeId,
                        routePrefix.blindingKey,
                        routePrefix.blindedNodes + destination.route.blindedNodes
                    )
                }
            }
        }
    }

    sealed class BuildMessageError
    data class MessageTooLarge(val payloadSize: Int) : BuildMessageError()

    /**
     * Builds an encrypted onion containing a message that should be relayed to the destination.
     *
     * @param sessionKey a random key to encrypt the onion.
     * @param blindingSecret a random key to create the blinded path.
     * @param intermediateNodes list of intermediate nodes between us and the destination (can be empty if we want to contact the destination directly).
     * @param destination the destination of this message, can be a node id or a blinded route.
     * @param content list of TLVs to send to the recipient of the message.
     */
    fun buildMessage(
        sessionKey: PrivateKey,
        blindingSecret: PrivateKey,
        intermediateNodes: List,
        destination: Destination,
        content: TlvStream
    ): Either {
        val route = buildRoute(blindingSecret, intermediateNodes, destination)
        val payloads = buildList {
            // Intermediate nodes only receive blinded path relay information.
            addAll(route.encryptedPayloads.dropLast(1).map { MessageOnion(TlvStream(OnionMessagePayloadTlv.EncryptedData(it))).write() })
            // The destination receives the message contents and the blinded path information.
            add(MessageOnion(content.copy(records = content.records + OnionMessagePayloadTlv.EncryptedData(route.encryptedPayloads.last()))).write())
        }
        val payloadSize = payloads.sumOf { it.size + Sphinx.MacLength }
        val packetSize = when {
            payloadSize <= 1300 -> 1300
            payloadSize <= 32768 -> 32768
            payloadSize <= 65432 -> 65432 // this corresponds to a total lightning message size of 65535
            else -> return Either.Left(MessageTooLarge(payloadSize))
        }
        // Since we are setting the packet size based on the payload, the onion creation should never fail.
        val packet = Sphinx.create(
            sessionKey,
            route.blindedNodes.map { it.blindedPublicKey },
            payloads,
            associatedData = null,
            packetSize
        ).packet
        return Either.Right(OnionMessage(route.blindingKey, packet))
    }

    /**
     * @param content message received.
     * @param blindedPrivateKey private key of the blinded node id used in our blinded path.
     * @param pathId path_id that we included in our blinded path for ourselves.
     */
    data class DecryptedMessage(val content: MessageOnion, val blindedPrivateKey: PrivateKey, val pathId: ByteVector)

    fun decryptMessage(privateKey: PrivateKey, msg: OnionMessage, logger: MDCLogger): DecryptedMessage? {
        val blindedPrivateKey = RouteBlinding.derivePrivateKey(privateKey, msg.blindingKey)
        return when (val decrypted = Sphinx.peel(blindedPrivateKey, associatedData = ByteVector.empty, msg.onionRoutingPacket)) {
            is Either.Right -> {
                val message = try {
                    MessageOnion.read(decrypted.value.payload.toByteArray())
                } catch (e: Throwable) {
                    logger.warning { "ignoring onion message that couldn't be decoded: ${e.message}" }
                    return null
                }
                when (val payload = RouteBlinding.decryptPayload(privateKey, msg.blindingKey, message.encryptedData)) {
                    is Either.Left -> {
                        logger.warning { "ignoring onion message that couldn't be decrypted: ${payload.value}" }
                        null
                    }
                    is Either.Right -> {
                        val (decryptedPayload, nextBlinding) = payload.value
                        when (val relayInfo = RouteBlindingEncryptedData.read(decryptedPayload.toByteArray())) {
                            is Either.Left -> {
                                logger.warning { "ignoring onion message with invalid relay info: ${relayInfo.value}" }
                                null
                            }
                            is Either.Right -> when {
                                !decrypted.value.isLastPacket && relayInfo.value.nextNodeId == EncodedNodeId.WithPublicKey.Wallet(privateKey.publicKey()) -> {
                                    // We may add ourselves to the route several times at the end to hide the real length of the route.
                                    val nextMessage = OnionMessage(relayInfo.value.nextBlindingOverride ?: nextBlinding, decrypted.value.nextPacket)
                                    decryptMessage(privateKey, nextMessage, logger)
                                }
                                decrypted.value.isLastPacket -> DecryptedMessage(message, blindedPrivateKey, relayInfo.value.pathId ?: ByteVector32.Zeroes)
                                else -> {
                                    logger.warning { "ignoring onion message for which we're not the destination (next_node_id=${relayInfo.value.nextNodeId}, path_id=${relayInfo.value.pathId?.toHex()})" }
                                    null
                                }
                            }
                        }
                    }
                }
            }
            is Either.Left -> {
                logger.warning { "ignoring onion message that couldn't be decrypted: ${decrypted.value.message}" }
                null
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy