From 51defce453c39444062615b06a59fab9da6fbbfb Mon Sep 17 00:00:00 2001 From: Thomas HUET <81159533+thomash-acinq@users.noreply.github.com> Date: Wed, 20 Nov 2024 12:07:23 +0100 Subject: [PATCH] Add logs for balance estimate (#2939) The balance estimates are sometimes outside the range [0, 1] which should never happen. We add some logs to help debug it. We also stop calling `TimestampSecond.now()` multiple times during the computation and reuse the same value which may fix the issue. --- .../acinq/eclair/router/BalanceEstimate.scala | 55 ++++---- .../eclair/router/BalanceEstimateSpec.scala | 132 +++++++++--------- 2 files changed, 99 insertions(+), 88 deletions(-) diff --git a/eclair-core/src/main/scala/fr/acinq/eclair/router/BalanceEstimate.scala b/eclair-core/src/main/scala/fr/acinq/eclair/router/BalanceEstimate.scala index 6aa56163bd..06a6fcd55d 100644 --- a/eclair-core/src/main/scala/fr/acinq/eclair/router/BalanceEstimate.scala +++ b/eclair-core/src/main/scala/fr/acinq/eclair/router/BalanceEstimate.scala @@ -16,6 +16,7 @@ package fr.acinq.eclair.router +import akka.event.LoggingAdapter import fr.acinq.bitcoin.scalacompat.Crypto.PublicKey import fr.acinq.bitcoin.scalacompat.{Satoshi, SatoshiLong} import fr.acinq.eclair.router.Graph.GraphStructure.{DirectedGraph, GraphEdge} @@ -108,8 +109,8 @@ case class BalanceEstimate private(low: MilliSatoshi, * @param t time at which we knew if we could or couldn't send amount * @return the probability that we can send amount now */ - private def decay(amount: MilliSatoshi, successProbabilityAtT: Double, t: TimestampSecond): Double = { - val decayRatio = 1 / math.pow(2, (TimestampSecond.now() - t) / halfLife) + private def decay(amount: MilliSatoshi, successProbabilityAtT: Double, t: TimestampSecond, now: TimestampSecond): Double = { + val decayRatio = 1 / math.pow(2, (now - t) / halfLife) val baseline = 1 - amount.toLong.toDouble / maxCapacity.toMilliSatoshi.toLong baseline * (1 - decayRatio) + successProbabilityAtT * decayRatio } @@ -123,7 +124,7 @@ case class BalanceEstimate private(low: MilliSatoshi, * or a spamming protection heuristic by the relaying node, but since we have no way of detecting that, our best * strategy is to ignore these cases. */ - def couldNotSend(amount: MilliSatoshi, timestamp: TimestampSecond): BalanceEstimate = { + def couldNotSend(amount: MilliSatoshi, timestamp: TimestampSecond)(implicit log: LoggingAdapter): BalanceEstimate = { if (amount <= low) { // the balance is actually below `low`, we discard our previous lower bound copy(low = 0 msat, lowTimestamp = timestamp, high = amount, highTimestamp = timestamp) @@ -135,7 +136,7 @@ case class BalanceEstimate private(low: MilliSatoshi, // was old enough that replacing it with the current amount decreases the success probability for `high`, then we // replace it. val updated = copy(high = amount, highTimestamp = timestamp) - if (updated.canSend(high) < this.canSend(high)) { + if (updated.canSend(high, timestamp) < this.canSend(high, timestamp)) { updated } else { this @@ -147,13 +148,13 @@ case class BalanceEstimate private(low: MilliSatoshi, * We tried to send the given amount, it was correctly relayed but failed afterwards, so we know we should be able to * send at least this amount again. */ - def couldSend(amount: MilliSatoshi, timestamp: TimestampSecond): BalanceEstimate = + def couldSend(amount: MilliSatoshi, timestamp: TimestampSecond)(implicit log: LoggingAdapter): BalanceEstimate = otherSide.couldNotSend(maxCapacity - amount, timestamp).otherSide /** * We successfully sent the given amount, so we know that some of the liquidity has shifted. */ - def didSend(amount: MilliSatoshi, timestamp: TimestampSecond): BalanceEstimate = { + def didSend(amount: MilliSatoshi, timestamp: TimestampSecond)(implicit log: LoggingAdapter): BalanceEstimate = { val newLow = (low - amount).max(0 msat) if (capacities.size == 1) { // Special case for single channel as we expect this case to be quite common and we can easily get more precise bounds. @@ -163,7 +164,7 @@ case class BalanceEstimate private(low: MilliSatoshi, // probability for the new high. val a = copy(low = newLow, high = newHigh) val b = copy(low = newLow, high = (maxCapacity - amount).max(0 msat), highTimestamp = timestamp) - if (a.canSend(newHigh) < b.canSend(newHigh)) { + if (a.canSend(newHigh, timestamp) < b.canSend(newHigh, timestamp)) { a } else { b @@ -176,7 +177,7 @@ case class BalanceEstimate private(low: MilliSatoshi, /** * We successfully received the given amount, so we know that some of the liquidity has shifted. */ - def didReceive(amount: MilliSatoshi, timestamp: TimestampSecond): BalanceEstimate = + def didReceive(amount: MilliSatoshi, timestamp: TimestampSecond)(implicit log: LoggingAdapter): BalanceEstimate = otherSide.didSend(amount, timestamp).otherSide def addEdge(edge: GraphEdge): BalanceEstimate = copy( @@ -203,17 +204,17 @@ case class BalanceEstimate private(low: MilliSatoshi, * - probability that it can relay a payment of high is decay(high, 0, highTimestamp) which is close to 0 if highTimestamp is recent * - probability that it can relay a payment of maxCapacity is 0 */ - def canSend(amount: MilliSatoshi): Double = { + def canSend(amount: MilliSatoshi, now: TimestampSecond)(implicit log: LoggingAdapter): Double = { val a = amount.toLong.toDouble val l = low.toLong.toDouble val h = high.toLong.toDouble val c = maxCapacity.toMilliSatoshi.toLong.toDouble // Success probability at the low and high points - val pLow = decay(low, 1, lowTimestamp) - val pHigh = decay(high, 0, highTimestamp) + val pLow = decay(low, 1, lowTimestamp, now) + val pHigh = decay(high, 0, highTimestamp, now) - if (amount < low) { + val estimate = if (amount < low) { (l - a * (1.0 - pLow)) / l } else if (amount < high) { ((h - a) * pLow + (a - l) * pHigh) / (h - l) @@ -222,6 +223,12 @@ case class BalanceEstimate private(low: MilliSatoshi, } else { 0 } + + if (estimate < 0 || estimate > 1) { + log.error("Could not estimate balance: this={}, amount={}, now={}", this, amount, now) + } + + estimate } } @@ -256,25 +263,25 @@ case class BalancesEstimates(balances: Map[(PublicKey, PublicKey), BalanceEstima defaultHalfLife ) - def channelCouldSend(hop: ChannelHop, amount: MilliSatoshi): BalancesEstimates = { + def channelCouldSend(hop: ChannelHop, amount: MilliSatoshi)(implicit log: LoggingAdapter): BalancesEstimates = { get(hop.nodeId, hop.nextNodeId).foreach { balance => - val estimatedProbability = balance.canSend(amount) + val estimatedProbability = balance.canSend(amount, TimestampSecond.now()) Monitoring.Metrics.remoteEdgeRelaySuccess(estimatedProbability) } BalancesEstimates(balances.updatedWith((hop.nodeId, hop.nextNodeId))(_.map(_.couldSend(amount, TimestampSecond.now()))), defaultHalfLife) } - def channelCouldNotSend(hop: ChannelHop, amount: MilliSatoshi): BalancesEstimates = { + def channelCouldNotSend(hop: ChannelHop, amount: MilliSatoshi)(implicit log: LoggingAdapter): BalancesEstimates = { get(hop.nodeId, hop.nextNodeId).foreach { balance => - val estimatedProbability = balance.canSend(amount) + val estimatedProbability = balance.canSend(amount, TimestampSecond.now()) Monitoring.Metrics.remoteEdgeRelayFailure(estimatedProbability) } BalancesEstimates(balances.updatedWith((hop.nodeId, hop.nextNodeId))(_.map(_.couldNotSend(amount, TimestampSecond.now()))), defaultHalfLife) } - def channelDidSend(hop: ChannelHop, amount: MilliSatoshi): BalancesEstimates = { + def channelDidSend(hop: ChannelHop, amount: MilliSatoshi)(implicit log: LoggingAdapter): BalancesEstimates = { get(hop.nodeId, hop.nextNodeId).foreach { balance => - val estimatedProbability = balance.canSend(amount) + val estimatedProbability = balance.canSend(amount, TimestampSecond.now()) Monitoring.Metrics.remoteEdgeRelaySuccess(estimatedProbability) } val balances1 = balances.updatedWith((hop.nodeId, hop.nextNodeId))(_.map(_.didSend(amount, TimestampSecond.now()))) @@ -298,7 +305,7 @@ case class GraphWithBalanceEstimates(graph: DirectedGraph, private val balances: descList.foldLeft(balances)((acc, edge) => acc.removeEdge(edge).removeEdge(edge.reversed)), ) - def routeCouldRelay(route: Route): GraphWithBalanceEstimates = { + def routeCouldRelay(route: Route)(implicit log: LoggingAdapter): GraphWithBalanceEstimates = { val (balances1, _) = route.hops.foldRight((balances, route.amount)) { case (hop, (balances, amount)) => (balances.channelCouldSend(hop, amount), amount + hop.fee(amount)) @@ -306,7 +313,7 @@ case class GraphWithBalanceEstimates(graph: DirectedGraph, private val balances: GraphWithBalanceEstimates(graph, balances1) } - def routeDidRelay(route: Route): GraphWithBalanceEstimates = { + def routeDidRelay(route: Route)(implicit log: LoggingAdapter): GraphWithBalanceEstimates = { val (balances1, _) = route.hops.foldRight((balances, route.amount)) { case (hop, (balances, amount)) => (balances.channelDidSend(hop, amount), amount + hop.fee(amount)) @@ -314,14 +321,14 @@ case class GraphWithBalanceEstimates(graph: DirectedGraph, private val balances: GraphWithBalanceEstimates(graph, balances1) } - def channelCouldNotSend(hop: ChannelHop, amount: MilliSatoshi): GraphWithBalanceEstimates = { + def channelCouldNotSend(hop: ChannelHop, amount: MilliSatoshi)(implicit log: LoggingAdapter): GraphWithBalanceEstimates = { GraphWithBalanceEstimates(graph, balances.channelCouldNotSend(hop, amount)) } - def canSend(amount: MilliSatoshi, edge: GraphEdge): Double = { + def canSend(amount: MilliSatoshi, edge: GraphEdge)(implicit log: LoggingAdapter): Double = { balances.balances.get((edge.desc.a, edge.desc.b)) match { - case Some(estimate) => estimate.canSend(amount) - case None => BalanceEstimate.empty(1 hour).addEdge(edge).canSend(amount) + case Some(estimate) => estimate.canSend(amount, TimestampSecond.now()) + case None => BalanceEstimate.empty(1 hour).addEdge(edge).canSend(amount, TimestampSecond.now()) } } } diff --git a/eclair-core/src/test/scala/fr/acinq/eclair/router/BalanceEstimateSpec.scala b/eclair-core/src/test/scala/fr/acinq/eclair/router/BalanceEstimateSpec.scala index 9784f93345..2f99ed3832 100644 --- a/eclair-core/src/test/scala/fr/acinq/eclair/router/BalanceEstimateSpec.scala +++ b/eclair-core/src/test/scala/fr/acinq/eclair/router/BalanceEstimateSpec.scala @@ -29,6 +29,8 @@ import scala.concurrent.duration.DurationInt class BalanceEstimateSpec extends AnyFunSuite { + implicit val log: akka.event.LoggingAdapter = akka.event.NoLogging + def isValid(balance: BalanceEstimate): Boolean = { balance.low >= 0.msat && balance.low <= balance.high && @@ -46,14 +48,15 @@ class BalanceEstimateSpec extends AnyFunSuite { test("no balance information") { val balance = BalanceEstimate.empty(1 day).addEdge(makeEdge(0, 100 sat)) + val now = TimestampSecond.now() assert(isValid(balance)) - assert(balance.canSend(0 msat) === 1.0 +- 0.001) - assert(balance.canSend(1 msat) === 1.0 +- 0.001) - assert(balance.canSend(25000 msat) === 0.75 +- 0.001) - assert(balance.canSend(50000 msat) === 0.5 +- 0.001) - assert(balance.canSend(75000 msat) === 0.25 +- 0.001) - assert(balance.canSend(99999 msat) === 0.0 +- 0.001) - assert(balance.canSend(100000 msat) === 0.0 +- 0.001) + assert(balance.canSend(0 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(1 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(25000 msat, now) === 0.75 +- 0.001) + assert(balance.canSend(50000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(75000 msat, now) === 0.25 +- 0.001) + assert(balance.canSend(99999 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(100000 msat, now) === 0.0 +- 0.001) } test("add and remove channels") { @@ -92,15 +95,15 @@ class BalanceEstimateSpec extends AnyFunSuite { .addEdge(makeEdge(1, 100 sat)) .couldSend(24000 msat, now) .couldNotSend(30000 msat, now) - assert(balance.canSend(0 msat) === 1.0 +- 0.001) - assert(balance.canSend(23999 msat) === 1.0 +- 0.001) - assert(balance.canSend(24000 msat) === 1.0 +- 0.001) - assert(balance.canSend(24001 msat) === 1.0 +- 0.001) - assert(balance.canSend(27000 msat) === 0.5 +- 0.001) - assert(balance.canSend(29999 msat) === 0.0 +- 0.001) - assert(balance.canSend(30000 msat) === 0.0 +- 0.001) - assert(balance.canSend(30001 msat) === 0.0 +- 0.001) - assert(balance.canSend(100000 msat) === 0.0 +- 0.001) + assert(balance.canSend(0 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(23999 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(24000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(24001 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(27000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(29999 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(30000 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(30001 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(100000 msat, now) === 0.0 +- 0.001) } test("update bounds based on what could then could not be sent (decreasing amounts)") { @@ -112,13 +115,13 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldSend(26000 msat, now) .couldNotSend(14000 msat, now) assert(isValid(balance)) - assert(balance.canSend(0 msat) === 1.0 +- 0.001) - assert(balance.canSend(1 msat) === 1.0 +- 0.001) - assert(balance.canSend(7000 msat) === 0.5 +- 0.001) - assert(balance.canSend(14000 msat) === 0.0 +- 0.001) - assert(balance.canSend(26000 msat) === 0.0 +- 0.001) - assert(balance.canSend(99999 msat) === 0.0 +- 0.001) - assert(balance.canSend(100000 msat) === 0.0 +- 0.001) + assert(balance.canSend(0 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(1 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(7000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(14000 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(26000 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(99999 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(100000 msat, now) === 0.0 +- 0.001) } test("update bounds based on what could not then could be sent (increasing amounts)") { @@ -130,12 +133,12 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldNotSend(26000 msat, now) .couldSend(30000 msat, now) assert(isValid(balance)) - assert(balance.canSend(0 msat) === 1.0 +- 0.001) - assert(balance.canSend(1 msat) === 1.0 +- 0.001) - assert(balance.canSend(30000 msat) === 1.0 +- 0.001) - assert(balance.canSend(65000 msat) === 0.5 +- 0.001) - assert(balance.canSend(99999 msat) === 0.0 +- 0.001) - assert(balance.canSend(100000 msat) === 0.0 +- 0.001) + assert(balance.canSend(0 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(1 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(30000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(65000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(99999 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(100000 msat, now) === 0.0 +- 0.001) } test("update bounds based on what could not then could be sent (decreasing amounts)") { @@ -147,26 +150,27 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldNotSend(30000 msat, now) .couldSend(20000 msat, now) assert(isValid(balance)) - assert(balance.canSend(0 msat) === 1.0 +- 0.001) - assert(balance.canSend(1 msat) === 1.0 +- 0.001) - assert(balance.canSend(20000 msat) === 1.0 +- 0.001) - assert(balance.canSend(25000 msat) === 0.5 +- 0.001) - assert(balance.canSend(30000 msat) === 0.0 +- 0.001) - assert(balance.canSend(99999 msat) === 0.0 +- 0.001) - assert(balance.canSend(100000 msat) === 0.0 +- 0.001) + assert(balance.canSend(0 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(1 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(20000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(25000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(30000 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(99999 msat, now) === 0.0 +- 0.001) + assert(balance.canSend(100000 msat, now) === 0.0 +- 0.001) } test("decay restores baseline bounds") { - val longAgo = TimestampSecond.now() - 30.seconds + val now = TimestampSecond.now() + val longAgo = now - 30.seconds val balance = BalanceEstimate.empty(1 second) .addEdge(makeEdge(0, 100 sat)) .couldNotSend(32000 msat, longAgo) .couldSend(28000 msat, longAgo) assert(isValid(balance)) - assert(balance.canSend(1 msat) === 1.0 +- 0.01) - assert(balance.canSend(33333 msat) === 0.666 +- 0.01) - assert(balance.canSend(66666 msat) === 0.333 +- 0.01) - assert(balance.canSend(99999 msat) === 0.0 +- 0.01) + assert(balance.canSend(1 msat, now) === 1.0 +- 0.01) + assert(balance.canSend(33333 msat, now) === 0.666 +- 0.01) + assert(balance.canSend(66666 msat, now) === 0.333 +- 0.01) + assert(balance.canSend(99999 msat, now) === 0.0 +- 0.01) } test("sending on single channel shifts amounts") { @@ -176,13 +180,13 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldNotSend(80000 msat, now) .couldSend(50000 msat, now) assert(isValid(balance)) - assert(balance.canSend(50000 msat) === 1.0 +- 0.001) - assert(balance.canSend(80000 msat) === 0.0 +- 0.001) + assert(balance.canSend(50000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(80000 msat, now) === 0.0 +- 0.001) val balanceAfterSend = balance.didSend(20000 msat, now) assert(isValid(balanceAfterSend)) - assert(balanceAfterSend.canSend(30000 msat) === 1.0 +- 0.001) - assert(balanceAfterSend.canSend(45000 msat) === 0.5 +- 0.001) - assert(balanceAfterSend.canSend(60000 msat) === 0.0 +- 0.001) + assert(balanceAfterSend.canSend(30000 msat, now) === 1.0 +- 0.001) + assert(balanceAfterSend.canSend(45000 msat, now) === 0.5 +- 0.001) + assert(balanceAfterSend.canSend(60000 msat, now) === 0.0 +- 0.001) } test("sending on single channel after decay") { @@ -194,10 +198,10 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldSend(50000 msat, longAgo) .didSend(40000 msat, now) assert(isValid(balance)) - assert(balance.canSend(0 msat) === 1.0 +- 0.01) - assert(balance.canSend(10000 msat) <= 0.9) - assert(balance.canSend(50000 msat) >= 0.1) - assert(balance.canSend(60000 msat) === 0.0 +- 0.01) + assert(balance.canSend(0 msat, now) === 1.0 +- 0.01) + assert(balance.canSend(10000 msat, now) <= 0.9) + assert(balance.canSend(50000 msat, now) >= 0.1) + assert(balance.canSend(60000 msat, now) === 0.0 +- 0.01) } test("sending on parallel channels shifts low only") { @@ -208,12 +212,12 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldNotSend(80000 msat, now) .couldSend(50000 msat, now) assert(isValid(balance)) - assert(balance.canSend(50000 msat) === 1.0 +- 0.001) - assert(balance.canSend(80000 msat) === 0.0 +- 0.001) + assert(balance.canSend(50000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(80000 msat, now) === 0.0 +- 0.001) val balanceAfterSend = balance.didSend(20000 msat, now) assert(isValid(balanceAfterSend)) - assert(balanceAfterSend.canSend(30000 msat) === 1.0 +- 0.001) - assert(balanceAfterSend.canSend(70000 msat) > 0.1) + assert(balanceAfterSend.canSend(30000 msat, now) === 1.0 +- 0.001) + assert(balanceAfterSend.canSend(70000 msat, now) > 0.1) } test("receiving on single channel shifts amounts") { @@ -224,9 +228,9 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldSend(50000 msat, now) .didReceive(10000 msat, now) assert(isValid(balance)) - assert(balance.canSend(60000 msat) === 1.0 +- 0.001) - assert(balance.canSend(75000 msat) === 0.5 +- 0.001) - assert(balance.canSend(90000 msat) === 0.0 +- 0.001) + assert(balance.canSend(60000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(75000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(90000 msat, now) === 0.0 +- 0.001) } test("receiving on single channel after decay") { @@ -238,10 +242,10 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldSend(50000 msat, longAgo) .didReceive(10000 msat, now) assert(isValid(balance)) - assert(balance.canSend(10000 msat) >= 0.9) - assert(balance.canSend(20000 msat) <= 0.9) - assert(balance.canSend(80000 msat) >= 0.1) - assert(balance.canSend(90000 msat) <= 0.1) + assert(balance.canSend(10000 msat, now) >= 0.9) + assert(balance.canSend(20000 msat, now) <= 0.9) + assert(balance.canSend(80000 msat, now) >= 0.1) + assert(balance.canSend(90000 msat, now) <= 0.1) } test("receiving on parallel channels shifts high only") { @@ -253,9 +257,9 @@ class BalanceEstimateSpec extends AnyFunSuite { .couldSend(50000 msat, now) .didReceive(20000 msat, now) assert(isValid(balance)) - assert(balance.canSend(50000 msat) === 1.0 +- 0.001) - assert(balance.canSend(70000 msat) === 0.5 +- 0.001) - assert(balance.canSend(90000 msat) === 0.0 +- 0.001) + assert(balance.canSend(50000 msat, now) === 1.0 +- 0.001) + assert(balance.canSend(70000 msat, now) === 0.5 +- 0.001) + assert(balance.canSend(90000 msat, now) === 0.0 +- 0.001) } test("baseline from graph") {