Communicating fees
In the LSP model, fees are involved when the user wants to receive a payment, but doesn't have a sufficient receivable amount. This section provides recommendations on how to communicate these fees to a user.
Before receiving a Lightning payment
When the user wants to receive a payment, a setup fee is paid when the resulting invoice would exceed the receivable amount. The setup fee is made up of two parts:
- A minimum fee
- A proportional fee based on the amount
Before creating an invoice, the amount the user will want to receive is yet unknown. It is recommended to show the user a message consisting of the following information:
A setup fee of x% with a minimum of y sats will be applied for receiving more than z sats.
You can construct this message as follows:
let inbound_liquidity_msat = sdk.node_info()?.max_receivable_single_payment_amount_msat;
let inbound_liquidity_sat = inbound_liquidity_msat / 1000;
let opening_fee_response = sdk
.open_channel_fee(OpenChannelFeeRequest::default())
.await?;
let opening_fees = opening_fee_response.fee_params;
let fee_percentage = (opening_fees.proportional * 100) as f64 / 1_000_000_f64;
let min_fee_sat = opening_fees.min_msat / 1_000;
if inbound_liquidity_sat == 0 {
info!("A setup fee of {fee_percentage}% with a minimum of {min_fee_sat} sats will be applied.");
} else {
info!("A setup fee of {fee_percentage}% with a minimum of {min_fee_sat} sats will be applied for receiving more than {inbound_liquidity_sat} sats.");
}
if let nodeInfo = try? sdk.nodeInfo() {
let inboundLiquiditySat = nodeInfo.maxReceivableSinglePaymentAmountMsat / 1_000;
let openingFeeResponse = try? sdk.openChannelFee(req: OpenChannelFeeRequest(amountMsat: nil));
if let openingFees = openingFeeResponse?.feeParams {
let feePercentage = Double(openingFees.proportional * 100) / 1_000_000.0;
let minFeeSat = openingFees.minMsat / 1_000;
if inboundLiquiditySat == 0 {
print("A setup fee of \(feePercentage)% with a minimum of \(minFeeSat) sats will be applied.");
} else {
print("A setup fee of \(feePercentage)% with a minimum of \(minFeeSat) sats will be applied for receiving more than \(inboundLiquiditySat) sats.");
}
}
}
val inboundLiquidityMsat = sdk.nodeInfo()?.maxReceivableSinglePaymentAmountMsat ?: 0u
val inboundLiquiditySat = inboundLiquidityMsat / 1_000u
val openingFeeResponse = sdk.openChannelFee(OpenChannelFeeRequest(null))
val openingFees = openingFeeResponse.feeParams;
val feePercentage = (openingFees.proportional * 100u) / 1_000_000u
val minFeeSat = openingFees.minMsat / 1_000u;
if (inboundLiquiditySat == 0UL) {
// Log.v("A setup fee of ${feePercentage}% with a minimum of ${minFeeSat} sats will be applied.");
} else {
// Log.v("A setup fee of ${feePercentage}% with a minimum of ${minFeeSat} sats will be applied for receiving more than ${inboundLiquiditySat} sats.");
}
const nodeState = await nodeInfo()
const inboundLiquidityMsat = nodeState.maxReceivableSinglePaymentAmountMsat
const inboundLiquiditySat = inboundLiquidityMsat != null ? inboundLiquidityMsat / 1_000 : 0
const openChannelFeeResponse = await openChannelFee({})
const openingFees = openChannelFeeResponse.feeParams
const feePercentage = (openingFees.proportional * 100) / 1_000_000
const minFeeSat = openingFees.minMsat / 1_000
if (inboundLiquiditySat === 0) {
console.log(`A setup fee of ${feePercentage}% with a minimum of ${minFeeSat} sats will be applied.`)
} else {
console.log(`A setup fee of ${feePercentage}% with a minimum of ${minFeeSat} sats will be applied for receiving more than ${inboundLiquiditySat} sats.`)
}
NodeState? nodeInfo = await breezSDK.nodeInfo();
if (nodeInfo != null) {
int inboundLiquiditySat = nodeInfo.maxReceivableSinglePaymentAmountMsat ~/ 1000;
OpenChannelFeeResponse openingFeeResponse = await breezSDK.openChannelFee(req: OpenChannelFeeRequest());
OpeningFeeParams openingFees = openingFeeResponse.feeParams;
double feePercentage = (openingFees.proportional * 100) / 1000000;
int minFeeSat = openingFees.minMsat ~/ 1000;
if (inboundLiquiditySat == 0) {
print("A setup fee of $feePercentage% with a minimum of $minFeeSat sats will be applied.");
} else {
print("A setup fee of $feePercentage% with a minimum of $minFeeSat sats will be applied for receiving more than $inboundLiquiditySat sats.");
}
}
inbound_liquidity_msat = sdk_services.node_info().max_receivable_single_payment_amount_msat
inbound_liquidity_sat = inbound_liquidity_msat / 1_000
opening_fee_response = sdk_services.open_channel_fee()
opening_fees = opening_fee_response.fee_params
fee_percentage = (opening_fees.proportional * 100) / 1_000_000
min_fee_sat = opening_fees.min_msat / 1_000
if inbound_liquidity_sat == 0:
print("A setup fee of ", fee_percentage, "% with a minimum of ", min_fee_sat, " sats will be applied.")
else:
print(
"A setup fee of ", fee_percentage, "% with a minimum of ", min_fee_sat, " sats will be applied "
"for receiving more than ", inbound_liquidity_sat, " sats."
)
nodeInfo, err := sdk.NodeInfo()
if err != nil {
return err
}
var inboundLiquiditySat = nodeInfo.MaxReceivableSinglePaymentAmountMsat / 1_000
openingFeeResponse, err := sdk.OpenChannelFee(breez_sdk.OpenChannelFeeRequest{})
if err != nil {
return err
}
var openingFees = openingFeeResponse.FeeParams
var feePercentage = (openingFees.Proportional * 100) / 1_000_000.0
var minFeeSat = openingFees.MinMsat / 1_000
if inboundLiquiditySat == 0 {
log.Printf(
"A setup fee of %v%% with a minimum of %v sats will be applied.",
feePercentage, minFeeSat)
} else {
log.Printf(
"A setup fee of %v%% with a minimum of %v sats will be applied"+
"for receiving more than %v sats.",
feePercentage, minFeeSat, inboundLiquiditySat)
}
try
{
var nodeInfo = sdk.NodeInfo();
var inboundLiquiditySat = nodeInfo?.maxReceivableSinglePaymentAmountMsat / 1_000;
var openingFeeResponse = sdk.OpenChannelFee(new OpenChannelFeeRequest(null));
var openingFees = openingFeeResponse?.feeParams;
if (openingFees != null)
{
var feePercentage = (openingFees.proportional * 100) / 1_000_000.0;
var minFeeSat = openingFees.minMsat / 1_000;
if (inboundLiquiditySat == 0)
{
Console.WriteLine(
$"A setup fee of {feePercentage}% with a minimum of {minFeeSat} sats will be applied."
);
}
else
{
Console.WriteLine(
$"A setup fee of {feePercentage}% with a minimum of {minFeeSat} sats will be applied " +
$"for receiving more than {inboundLiquiditySat} sats."
);
}
}
}
catch (Exception)
{
// Handle error
}
After creating an invoice
After calling receive_payment
, you would typically show the recipient a screen containing a QR code with the invoice that the sender can scan.
This is another place to show the user the opening fees applied to the invoice. At this point the amount the user wants to receive is known, so the message can be more concise:
A setup fee of x sats is applied to this invoice.
The fiat amount can also be included. In case of a mobile app it is recommended to communicate to the user that the app has to be run in the foreground in order to be able to receive the payment.
Developer note
Consider implementing the Notification Plugin when using the Breez SDK in a mobile application. By registering a webhook the application can receive a payment notification to process the payment in the background.Here is how you can build this message:
let opening_fee_sat = receive_payment_response
.opening_fee_msat
.unwrap_or_default()
/ 1000;
info!("A setup fee of {opening_fee_sat} sats is applied to this invoice.");
let openingFeeSat = (receivePaymentResponse.openingFeeMsat ?? 0) / 1_000;
print("A setup fee of \(openingFeeSat) sats is applied to this invoice.")
val openingFeeSat = receivePaymentResponse.openingFeeMsat?.let{ it.div(1000UL).toULong() } ?: run { 0UL }
// Log.v("Breez", "A setup fee of ${openingFeeSat} sats is applied to this invoice.")
const openingFeeMsat = receivePaymentResponse.openingFeeMsat
const openingFeeSat = openingFeeMsat != null ? openingFeeMsat / 1000 : 0
console.log(`A setup fee of ${openingFeeSat} sats is applied to this invoice.`)
int openingFeeSat = (receivePaymentResponse.openingFeeMsat ?? 0) / 1000 as int;
print("A setup fee of $openingFeeSat sats is applied to this invoice.");
opening_fee_msat = receive_payment_response.opening_fee_msat
opening_fee_sat = opening_fee_msat / 1_000 if opening_fee_msat is not None else 0
print("A setup fee of ", opening_fee_sat, " sats is applied to this invoice.")
var openingFeeSat int
openingFeeMsat := receivePaymentResponse.OpeningFeeMsat
if openingFeeMsat != nil {
openingFeeSat = int(*openingFeeMsat / 1_000)
}
log.Printf("A setup fee of %v sats is applied to this invoice.", openingFeeSat)
var openingFeeSat = receivePaymentResponse.openingFeeMsat.GetValueOrDefault() / 1000;
Console.WriteLine($"A setup fee of {openingFeeSat} sats is applied to this invoice.");
Sending a Lightning payment
Routing fees for sending Lightning payments vary based on the available path, as some channels incur higher fees due to the distribution of liquidity across the network. Developers have the option to set limits on Lightning fees, which are capped by default at 1% of the payment amount. However, it's important to note that restricting fees can increase the likelihood of payment failures.
For users leveraging trampoline payments (recommended for reliabily), routing fees are currently fixed at 0.5%.
Receiving an on-chain transaction
For receiving onchain, there is a minimum and a maximum amount the user can receive. The fees are made up of the same components as receiving a lightning payment.
The user gets an onchain address from receive_onchain
. There is no way to know ahead of time exactly the amount that will be received on this address, so it is recommended to show the user the receivable boundaries and the fees involved:
Send more than v sats and up to w sats to this address. A setup fee of x% with a minimum of y sats will be applied for sending more than z sats. This address can only be used once.
Below code sample constructs this message.
let swap_info = sdk
.receive_onchain(ReceiveOnchainRequest::default())
.await?;
let min_deposit_sat = swap_info.min_allowed_deposit;
let max_deposit_sat = swap_info.max_allowed_deposit;
let inbound_liquidity_sat = sdk.node_info()?.max_receivable_single_payment_amount_msat / 1000;
if let Some(swap_opening_fees) = swap_info.channel_opening_fees {
let fee_percentage = (swap_opening_fees.proportional * 100) as f64 / 1_000_000_f64;
let min_fee_sat = swap_opening_fees.min_msat / 1_000;
info!("Send more than {min_deposit_sat} sats and up to {max_deposit_sat} sats to this address. \
A setup fee of {fee_percentage}% with a minimum of {min_fee_sat} sats will be applied \
for sending more than {inbound_liquidity_sat} sats. This address can only be used once.");
}
let swapInfo = try? sdk.receiveOnchain(req: ReceiveOnchainRequest())
let minDepositSat = swapInfo!.minAllowedDeposit;
let maxDepositSat = swapInfo!.maxAllowedDeposit;
let nodeInfo = try? sdk.nodeInfo();
let inboundLiquiditySat = nodeInfo!.maxReceivableSinglePaymentAmountMsat / 1_000;
if let swapOpeningFees = swapInfo!.channelOpeningFees {
let feePercentage = Double(swapOpeningFees.proportional * 100) / 1_000_000.0;
let minFeeSat = swapOpeningFees.minMsat / 1_000;
print("Send more than \(minDepositSat) sats and up to \(maxDepositSat) sats to this address. A setup fee of \(feePercentage)% with a minimum of \(minFeeSat) sats will be applied for sending more than \(inboundLiquiditySat) sats. This address can only be used once.");
}
val swapInfo = sdk.receiveOnchain(ReceiveOnchainRequest())
val minDepositSat = swapInfo.minAllowedDeposit
val maxDepositSat = swapInfo.maxAllowedDeposit
val inboundLiquiditySat = (sdk.nodeInfo()?.maxReceivableSinglePaymentAmountMsat ?: 0u) / 1_000u
val swapOpeningFees = swapInfo.channelOpeningFees
if (swapOpeningFees != null) {
val feePercentage = (swapOpeningFees.proportional * 100u) / 1_000_000u
val minFeeSat = swapOpeningFees.minMsat / 1_000u;
// Log.v("Send more than ${minDepositSat} sats and up to ${maxDepositSat} sats to this address. A setup fee of ${feePercentage}% with a minimum of ${minFeeSat} sats will be applied for sending more than ${inboundLiquiditySat} sats. This address can only be used once.");
}
const nodeState = await nodeInfo()
const swapInfo = await receiveOnchain({})
const minDepositSat = swapInfo.minAllowedDeposit
const maxDepositSat = swapInfo.maxAllowedDeposit
const inboundLiquidityMsat = nodeState?.maxReceivableSinglePaymentAmountMsat
const inboundLiquiditySat = inboundLiquidityMsat != null ? (inboundLiquidityMsat / 1_000) : 0
const swapOpeningFees = swapInfo.channelOpeningFees
if (swapOpeningFees != null) {
const feePercentage = (swapOpeningFees.proportional * 100) / 1_000_000
const minFeeSat = swapOpeningFees.minMsat / 1_000
console.log(`Send more than ${minDepositSat} sats and up to ${maxDepositSat} sats to this address. A setup fee of ${feePercentage}% with a minimum of ${minFeeSat} sats will be applied for sending more than ${inboundLiquiditySat} sats. This address can only be used once.`)
}
SwapInfo swapInfo = await breezSDK.receiveOnchain(req: ReceiveOnchainRequest());
int minDepositSat = swapInfo.minAllowedDeposit;
int maxDepositSat = swapInfo.maxAllowedDeposit;
NodeState? nodeInfo = await breezSDK.nodeInfo();
if (nodeInfo != null) {
int inboundLiquiditySat = nodeInfo.maxReceivableSinglePaymentAmountMsat ~/ 1000;
OpeningFeeParams? swapOpeningFees = swapInfo.channelOpeningFees;
if (swapOpeningFees != null) {
double feePercentage = (swapOpeningFees.proportional * 100) / 1000000;
int minFeeSat = swapOpeningFees.minMsat ~/ 1000;
print("Send more than $minDepositSat sats and up to $maxDepositSat sats to this address. A setup fee of $feePercentage% with a minimum of $minFeeSat sats will be applied for sending more than $inboundLiquiditySat sats. This address can only be used once.");
}
}
swap_info = sdk_services.receive_onchain(breez_sdk.ReceiveOnchainRequest())
min_deposit_sat = swap_info.min_allowed_deposit
max_deposit_sat = swap_info.max_allowed_deposit
inbound_liquidity_sat = sdk_services.node_info().max_receivable_single_payment_amount_msat / 1_000
swap_opening_fees = swap_info.channel_opening_fees
if swap_opening_fees is not None:
fee_percentage = (swap_opening_fees.proportional * 100) / 1_000_000
min_fee_sat = swap_opening_fees.min_msat / 1_000
print(
"Send more than ", min_deposit_sat, " sats and up to ", max_deposit_sat, " sats to this address. "
"A setup fee of ", fee_percentage, "% with a minimum of ", min_fee_sat, " sats will be applied for "
"sending more than ", inbound_liquidity_sat, " sats. This address can only be used once."
)
swapInfo, err := sdk.ReceiveOnchain(breez_sdk.ReceiveOnchainRequest{})
if err != nil {
return err
}
var minDepositSat = swapInfo.MinAllowedDeposit
var maxDepositSat = swapInfo.MaxAllowedDeposit
nodeInfo, err := sdk.NodeInfo()
if err != nil {
return err
}
var inboundLiquiditySat = nodeInfo.MaxReceivableSinglePaymentAmountMsat / 1_000
var swapOpeningFees = swapInfo.ChannelOpeningFees
var feePercentage = (swapOpeningFees.Proportional * 100) / 1_000_000.0
var minFeeSat = swapOpeningFees.MinMsat / 1_000
log.Printf(
"Send more than %v sats and up to %v sats to this address. "+
"A setup fee of %v%% with a minimum of %v sats will be applied "+
"for sending more than %v sats. This address can only be used once.",
minDepositSat, maxDepositSat, feePercentage, minFeeSat, inboundLiquiditySat)
try
{
var swapInfo = sdk.ReceiveOnchain(new ReceiveOnchainRequest());
var minDepositSat = swapInfo?.minAllowedDeposit;
var maxDepositSat = swapInfo?.maxAllowedDeposit;
var nodeInfo = sdk.NodeInfo();
var inboundLiquiditySat = nodeInfo?.maxReceivableSinglePaymentAmountMsat / 1_000;
var swapOpeningFees = swapInfo?.channelOpeningFees;
if (swapOpeningFees != null)
{
var feePercentage = (swapOpeningFees.proportional * 100) / 1_000_000.0;
var minFeeSat = swapOpeningFees.minMsat / 1_000;
Console.WriteLine(
$"Send more than {minDepositSat} sats and up to {maxDepositSat} sats to this address. " +
$"A setup fee of {feePercentage}% with a minimum of {minFeeSat} sats will be applied " +
$"for sending more than {inboundLiquiditySat} sats. This address can only be used once."
);
}
}
catch (Exception)
{
// Handle error
}
Sending an on-chain transaction
Sending an on-chain transaction to a BTC address involves a trustless reverse swap. Reverse swaps have a minimum transaction amount of 50,000 sats, which may increase during periods of high fees. The reverse swap provider, Boltz, charges a fixed service fee of 0.5% plus an additional mining fee based on current Bitcoin mempool usage.
Follow the instructions here to calculate limits and fees.