mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-01-18 05:12:45 +01:00
bcli: become less aggressive with onchain fee levels.
Users are more upset recently with the cost of unilateral closes than they are the risk of being cheated. While we complete our anchor implementation so we can use low fees there, let's get less aggressive (we already have 34 or 18 blocks to close in the worst case). The changes are: - Commit transactions were "2 CONSERVATIVE" now "6 ECONOMICAL". - HTLC resolution txs were "3 CONSERVATIVE" now "6 ECONOMICAL". - Penalty txs were "3 CONSERVATIVE" now "12 ECONOMICAL". - Normal txs were "4 ECONOMICAL" now "12 ECONOMICAL". There can be no perfect levels, but we have had understandable complaints recently about how high our default fee levels are. Changelog-Changed: Protocol: channel feerates reduced to bitcoind's "6 block ECONOMICAL" rate. Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
This commit is contained in:
parent
9a22e7b3a1
commit
d8e68893f5
@ -1014,9 +1014,9 @@ class LightningNode(object):
|
|||||||
params = r['params']
|
params = r['params']
|
||||||
if params == [2, 'CONSERVATIVE']:
|
if params == [2, 'CONSERVATIVE']:
|
||||||
feerate = feerates[0] * 4
|
feerate = feerates[0] * 4
|
||||||
elif params == [3, 'CONSERVATIVE']:
|
elif params == [6, 'ECONOMICAL']:
|
||||||
feerate = feerates[1] * 4
|
feerate = feerates[1] * 4
|
||||||
elif params == [4, 'ECONOMICAL']:
|
elif params == [12, 'ECONOMICAL']:
|
||||||
feerate = feerates[2] * 4
|
feerate = feerates[2] * 4
|
||||||
elif params == [100, 'ECONOMICAL']:
|
elif params == [100, 'ECONOMICAL']:
|
||||||
feerate = feerates[3] * 4
|
feerate = feerates[3] * 4
|
||||||
|
@ -410,7 +410,7 @@ static struct command_result *process_getblockchaininfo(struct bitcoin_cli *bcli
|
|||||||
}
|
}
|
||||||
|
|
||||||
enum feerate_levels {
|
enum feerate_levels {
|
||||||
FEERATE_VERY_URGENT,
|
FEERATE_HIGHEST,
|
||||||
FEERATE_URGENT,
|
FEERATE_URGENT,
|
||||||
FEERATE_NORMAL,
|
FEERATE_NORMAL,
|
||||||
FEERATE_SLOW,
|
FEERATE_SLOW,
|
||||||
@ -604,19 +604,15 @@ static struct command_result *getchaininfo(struct command *cmd,
|
|||||||
/* Mutual recursion. */
|
/* Mutual recursion. */
|
||||||
static struct command_result *estimatefees_done(struct bitcoin_cli *bcli);
|
static struct command_result *estimatefees_done(struct bitcoin_cli *bcli);
|
||||||
|
|
||||||
/*
|
|
||||||
* Calls `estimatesmartfee` with targets 2/CONSERVATIVE (very urgent),
|
|
||||||
* 3/CONSERVATIVE (urgent), 4/ECONOMICAL (normal), and 100/ECONOMICAL (slow)
|
|
||||||
*/
|
|
||||||
struct estimatefee_params {
|
struct estimatefee_params {
|
||||||
u32 blocks;
|
u32 blocks;
|
||||||
const char *style;
|
const char *style;
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct estimatefee_params estimatefee_params[] = {
|
static const struct estimatefee_params estimatefee_params[] = {
|
||||||
[FEERATE_VERY_URGENT] = { 2, "CONSERVATIVE" },
|
[FEERATE_HIGHEST] = { 2, "CONSERVATIVE" },
|
||||||
[FEERATE_URGENT] = { 3, "CONSERVATIVE" },
|
[FEERATE_URGENT] = { 6, "ECONOMICAL" },
|
||||||
[FEERATE_NORMAL] = { 4, "ECONOMICAL" },
|
[FEERATE_NORMAL] = { 12, "ECONOMICAL" },
|
||||||
[FEERATE_SLOW] = { 100, "ECONOMICAL" },
|
[FEERATE_SLOW] = { 100, "ECONOMICAL" },
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -640,10 +636,10 @@ static struct command_result *estimatefees_next(struct command *cmd,
|
|||||||
json_add_u64(response, "opening", stash->perkb[FEERATE_NORMAL]);
|
json_add_u64(response, "opening", stash->perkb[FEERATE_NORMAL]);
|
||||||
json_add_u64(response, "mutual_close", stash->perkb[FEERATE_SLOW]);
|
json_add_u64(response, "mutual_close", stash->perkb[FEERATE_SLOW]);
|
||||||
json_add_u64(response, "unilateral_close",
|
json_add_u64(response, "unilateral_close",
|
||||||
stash->perkb[FEERATE_VERY_URGENT] * bitcoind->commit_fee_percent / 100);
|
stash->perkb[FEERATE_URGENT] * bitcoind->commit_fee_percent / 100);
|
||||||
json_add_u64(response, "delayed_to_us", stash->perkb[FEERATE_NORMAL]);
|
json_add_u64(response, "delayed_to_us", stash->perkb[FEERATE_NORMAL]);
|
||||||
json_add_u64(response, "htlc_resolution", stash->perkb[FEERATE_URGENT]);
|
json_add_u64(response, "htlc_resolution", stash->perkb[FEERATE_URGENT]);
|
||||||
json_add_u64(response, "penalty", stash->perkb[FEERATE_URGENT]);
|
json_add_u64(response, "penalty", stash->perkb[FEERATE_NORMAL]);
|
||||||
/* We divide the slow feerate for the minimum acceptable, lightningd
|
/* We divide the slow feerate for the minimum acceptable, lightningd
|
||||||
* will use floor if it's hit, though. */
|
* will use floor if it's hit, though. */
|
||||||
json_add_u64(response, "min_acceptable",
|
json_add_u64(response, "min_acceptable",
|
||||||
@ -655,7 +651,7 @@ static struct command_result *estimatefees_next(struct command *cmd,
|
|||||||
* margin (say 5x the expected fee requirement)
|
* margin (say 5x the expected fee requirement)
|
||||||
*/
|
*/
|
||||||
json_add_u64(response, "max_acceptable",
|
json_add_u64(response, "max_acceptable",
|
||||||
stash->perkb[FEERATE_VERY_URGENT]
|
stash->perkb[FEERATE_HIGHEST]
|
||||||
* bitcoind->max_fee_multiplier);
|
* bitcoind->max_fee_multiplier);
|
||||||
return command_finished(cmd, response);
|
return command_finished(cmd, response);
|
||||||
}
|
}
|
||||||
|
@ -814,7 +814,7 @@ def test_penalty_htlc_tx_fulfill(node_factory, bitcoind, chainparams):
|
|||||||
|
|
||||||
# reconnect with l1, which will fulfill the payment
|
# reconnect with l1, which will fulfill the payment
|
||||||
l2.rpc.connect(l1.info['id'], 'localhost', l1.port)
|
l2.rpc.connect(l1.info['id'], 'localhost', l1.port)
|
||||||
l2.daemon.wait_for_log('got commitsig .*: feerate 15000, 0 added, 1 fulfilled, 0 failed, 0 changed')
|
l2.daemon.wait_for_log('got commitsig .*: feerate 11000, 0 added, 1 fulfilled, 0 failed, 0 changed')
|
||||||
l2.daemon.wait_for_log('coins payment_hash: {}'.format(sticky_inv['payment_hash']))
|
l2.daemon.wait_for_log('coins payment_hash: {}'.format(sticky_inv['payment_hash']))
|
||||||
|
|
||||||
# l2 moves on for closed l3
|
# l2 moves on for closed l3
|
||||||
@ -970,7 +970,7 @@ def test_penalty_htlc_tx_timeout(node_factory, bitcoind, chainparams):
|
|||||||
|
|
||||||
# reconnect with l1, which will fulfill the payment
|
# reconnect with l1, which will fulfill the payment
|
||||||
l2.rpc.connect(l1.info['id'], 'localhost', l1.port)
|
l2.rpc.connect(l1.info['id'], 'localhost', l1.port)
|
||||||
l2.daemon.wait_for_log('got commitsig .*: feerate 15000, 0 added, 1 fulfilled, 0 failed, 0 changed')
|
l2.daemon.wait_for_log('got commitsig .*: feerate 11000, 0 added, 1 fulfilled, 0 failed, 0 changed')
|
||||||
l2.daemon.wait_for_log('coins payment_hash: {}'.format(sticky_inv_2['payment_hash']))
|
l2.daemon.wait_for_log('coins payment_hash: {}'.format(sticky_inv_2['payment_hash']))
|
||||||
|
|
||||||
# l2 moves on for closed l3
|
# l2 moves on for closed l3
|
||||||
@ -2000,11 +2000,11 @@ def test_onchain_different_fees(node_factory, bitcoind, executor):
|
|||||||
# Both sides should have correct feerate
|
# Both sides should have correct feerate
|
||||||
assert l1.db_query('SELECT min_possible_feerate, max_possible_feerate FROM channels;') == [{
|
assert l1.db_query('SELECT min_possible_feerate, max_possible_feerate FROM channels;') == [{
|
||||||
'min_possible_feerate': 5000,
|
'min_possible_feerate': 5000,
|
||||||
'max_possible_feerate': 16000
|
'max_possible_feerate': 11000
|
||||||
}]
|
}]
|
||||||
assert l2.db_query('SELECT min_possible_feerate, max_possible_feerate FROM channels;') == [{
|
assert l2.db_query('SELECT min_possible_feerate, max_possible_feerate FROM channels;') == [{
|
||||||
'min_possible_feerate': 5000,
|
'min_possible_feerate': 5000,
|
||||||
'max_possible_feerate': 16000
|
'max_possible_feerate': 11000
|
||||||
}]
|
}]
|
||||||
|
|
||||||
bitcoind.generate_block(5)
|
bitcoind.generate_block(5)
|
||||||
|
@ -1969,7 +1969,7 @@ def test_update_fee(node_factory, bitcoind):
|
|||||||
# Make payments.
|
# Make payments.
|
||||||
l1.pay(l2, 200000000)
|
l1.pay(l2, 200000000)
|
||||||
# First payment causes fee update.
|
# First payment causes fee update.
|
||||||
l2.daemon.wait_for_log('peer updated fee to 14000')
|
l2.daemon.wait_for_log('peer updated fee to 11000')
|
||||||
l2.pay(l1, 100000000)
|
l2.pay(l1, 100000000)
|
||||||
|
|
||||||
# Now shutdown cleanly.
|
# Now shutdown cleanly.
|
||||||
@ -2049,7 +2049,7 @@ def test_fee_limits(node_factory, bitcoind):
|
|||||||
|
|
||||||
# Try stupid high fees
|
# Try stupid high fees
|
||||||
l1.stop()
|
l1.stop()
|
||||||
l1.set_feerates((15000 * 10, 11000, 7500, 3750), False)
|
l1.set_feerates((15000, 11000 * 10, 7500, 3750), False)
|
||||||
l1.start()
|
l1.start()
|
||||||
|
|
||||||
l3.daemon.wait_for_log('peer_in WIRE_UPDATE_FEE')
|
l3.daemon.wait_for_log('peer_in WIRE_UPDATE_FEE')
|
||||||
@ -2892,7 +2892,7 @@ def test_feerate_spam(node_factory, chainparams):
|
|||||||
l1.pay(l2, 10**9 - slack)
|
l1.pay(l2, 10**9 - slack)
|
||||||
|
|
||||||
# It will send this once (may have happened before line_graph's wait)
|
# It will send this once (may have happened before line_graph's wait)
|
||||||
wait_for(lambda: l1.daemon.is_in_log('Setting REMOTE feerate to 15000'))
|
wait_for(lambda: l1.daemon.is_in_log('Setting REMOTE feerate to 11000'))
|
||||||
wait_for(lambda: l1.daemon.is_in_log('peer_out WIRE_UPDATE_FEE'))
|
wait_for(lambda: l1.daemon.is_in_log('peer_out WIRE_UPDATE_FEE'))
|
||||||
|
|
||||||
# Now change feerates to something l1 can't afford.
|
# Now change feerates to something l1 can't afford.
|
||||||
|
@ -1470,54 +1470,50 @@ def test_feerates(node_factory):
|
|||||||
assert t not in feerates['perkb']
|
assert t not in feerates['perkb']
|
||||||
|
|
||||||
# Now try setting them, one at a time.
|
# Now try setting them, one at a time.
|
||||||
# Set CONSERVATIVE/2 feerate, for max and unilateral_close
|
# Set CONSERVATIVE/2 feerate, for max
|
||||||
l1.set_feerates((15000, 0, 0, 0), True)
|
l1.set_feerates((15000, 0, 0, 0), True)
|
||||||
wait_for(lambda: len(l1.rpc.feerates('perkw')['perkw']) == 3)
|
wait_for(lambda: len(l1.rpc.feerates('perkw')['perkw']) == 2)
|
||||||
feerates = l1.rpc.feerates('perkw')
|
feerates = l1.rpc.feerates('perkw')
|
||||||
assert feerates['perkw']['unilateral_close'] == 15000
|
|
||||||
assert feerates['warning_missing_feerates'] == 'Some fee estimates unavailable: bitcoind startup?'
|
assert feerates['warning_missing_feerates'] == 'Some fee estimates unavailable: bitcoind startup?'
|
||||||
assert 'perkb' not in feerates
|
assert 'perkb' not in feerates
|
||||||
assert feerates['perkw']['max_acceptable'] == 15000 * 10
|
assert feerates['perkw']['max_acceptable'] == 15000 * 10
|
||||||
assert feerates['perkw']['min_acceptable'] == 253
|
assert feerates['perkw']['min_acceptable'] == 253
|
||||||
|
|
||||||
# Set CONSERVATIVE/3 feerate, for htlc_resolution and penalty
|
# Set ECONOMICAL/6 feerate, for unilateral_close and htlc_resolution
|
||||||
l1.set_feerates((15000, 11000, 0, 0), True)
|
l1.set_feerates((15000, 11000, 0, 0), True)
|
||||||
wait_for(lambda: len(l1.rpc.feerates('perkw')['perkw']) == 5)
|
wait_for(lambda: len(l1.rpc.feerates('perkw')['perkw']) == 4)
|
||||||
feerates = l1.rpc.feerates('perkw')
|
feerates = l1.rpc.feerates('perkw')
|
||||||
assert feerates['perkw']['unilateral_close'] == 15000
|
assert feerates['perkw']['unilateral_close'] == 11000
|
||||||
assert feerates['perkw']['htlc_resolution'] == 11000
|
assert feerates['perkw']['htlc_resolution'] == 11000
|
||||||
assert feerates['perkw']['penalty'] == 11000
|
|
||||||
assert feerates['warning_missing_feerates'] == 'Some fee estimates unavailable: bitcoind startup?'
|
assert feerates['warning_missing_feerates'] == 'Some fee estimates unavailable: bitcoind startup?'
|
||||||
assert 'perkb' not in feerates
|
assert 'perkb' not in feerates
|
||||||
assert feerates['perkw']['max_acceptable'] == 15000 * 10
|
assert feerates['perkw']['max_acceptable'] == 15000 * 10
|
||||||
assert feerates['perkw']['min_acceptable'] == 253
|
assert feerates['perkw']['min_acceptable'] == 253
|
||||||
|
|
||||||
# Set ECONOMICAL/4 feerate, for all but min (so, no mutual_close feerate)
|
# Set ECONOMICAL/12 feerate, for all but min (so, no mutual_close feerate)
|
||||||
l1.set_feerates((15000, 11000, 6250, 0), True)
|
l1.set_feerates((15000, 11000, 6250, 0), True)
|
||||||
wait_for(lambda: len(l1.rpc.feerates('perkb')['perkb']) == len(types) - 1 + 2)
|
wait_for(lambda: len(l1.rpc.feerates('perkb')['perkb']) == len(types) - 1 + 2)
|
||||||
feerates = l1.rpc.feerates('perkb')
|
feerates = l1.rpc.feerates('perkb')
|
||||||
assert feerates['perkb']['unilateral_close'] == 15000 * 4
|
assert feerates['perkb']['unilateral_close'] == 11000 * 4
|
||||||
assert feerates['perkb']['htlc_resolution'] == 11000 * 4
|
assert feerates['perkb']['htlc_resolution'] == 11000 * 4
|
||||||
assert feerates['perkb']['penalty'] == 11000 * 4
|
|
||||||
assert 'mutual_close' not in feerates['perkb']
|
assert 'mutual_close' not in feerates['perkb']
|
||||||
for t in types:
|
for t in types:
|
||||||
if t not in ("unilateral_close", "htlc_resolution", "penalty", "mutual_close"):
|
if t not in ("unilateral_close", "htlc_resolution", "mutual_close"):
|
||||||
assert feerates['perkb'][t] == 25000
|
assert feerates['perkb'][t] == 25000
|
||||||
assert feerates['warning_missing_feerates'] == 'Some fee estimates unavailable: bitcoind startup?'
|
assert feerates['warning_missing_feerates'] == 'Some fee estimates unavailable: bitcoind startup?'
|
||||||
assert 'perkw' not in feerates
|
assert 'perkw' not in feerates
|
||||||
assert feerates['perkb']['max_acceptable'] == 15000 * 4 * 10
|
assert feerates['perkb']['max_acceptable'] == 15000 * 4 * 10
|
||||||
assert feerates['perkb']['min_acceptable'] == 253 * 4
|
assert feerates['perkb']['min_acceptable'] == 253 * 4
|
||||||
|
|
||||||
# Set ECONOMICAL/100 feerate for min
|
# Set ECONOMICAL/100 feerate for min and mutual_close
|
||||||
l1.set_feerates((15000, 11000, 6250, 5000), True)
|
l1.set_feerates((15000, 11000, 6250, 5000), True)
|
||||||
wait_for(lambda: len(l1.rpc.feerates('perkw')['perkw']) >= len(types) + 2)
|
wait_for(lambda: len(l1.rpc.feerates('perkw')['perkw']) >= len(types) + 2)
|
||||||
feerates = l1.rpc.feerates('perkw')
|
feerates = l1.rpc.feerates('perkw')
|
||||||
assert feerates['perkw']['unilateral_close'] == 15000
|
assert feerates['perkw']['unilateral_close'] == 11000
|
||||||
assert feerates['perkw']['htlc_resolution'] == 11000
|
assert feerates['perkw']['htlc_resolution'] == 11000
|
||||||
assert feerates['perkw']['penalty'] == 11000
|
|
||||||
assert feerates['perkw']['mutual_close'] == 5000
|
assert feerates['perkw']['mutual_close'] == 5000
|
||||||
for t in types:
|
for t in types:
|
||||||
if t not in ("unilateral_close", "htlc_resolution", "penalty", "mutual_close"):
|
if t not in ("unilateral_close", "htlc_resolution", "mutual_close"):
|
||||||
assert feerates['perkw'][t] == 25000 // 4
|
assert feerates['perkw'][t] == 25000 // 4
|
||||||
assert 'warning' not in feerates
|
assert 'warning' not in feerates
|
||||||
assert 'perkb' not in feerates
|
assert 'perkb' not in feerates
|
||||||
@ -2402,7 +2398,7 @@ def test_commitfee_option(node_factory):
|
|||||||
|
|
||||||
mock_wu = 5000
|
mock_wu = 5000
|
||||||
for l in [l1, l2]:
|
for l in [l1, l2]:
|
||||||
l.set_feerates((mock_wu, 0, 0, 0), True)
|
l.set_feerates((0, mock_wu, 0, 0), True)
|
||||||
l1_commit_fees = l1.rpc.call("estimatefees")["unilateral_close"]
|
l1_commit_fees = l1.rpc.call("estimatefees")["unilateral_close"]
|
||||||
l2_commit_fees = l2.rpc.call("estimatefees")["unilateral_close"]
|
l2_commit_fees = l2.rpc.call("estimatefees")["unilateral_close"]
|
||||||
|
|
||||||
|
@ -259,7 +259,7 @@ def test_pay_disconnect(node_factory, bitcoind):
|
|||||||
l1.daemon.wait_for_log('peer_out WIRE_CHANNEL_REESTABLISH')
|
l1.daemon.wait_for_log('peer_out WIRE_CHANNEL_REESTABLISH')
|
||||||
|
|
||||||
# Make l2 upset by asking for crazy fee.
|
# Make l2 upset by asking for crazy fee.
|
||||||
l1.set_feerates((10**6, 1000**6, 1000**6, 1000**6), False)
|
l1.set_feerates((10**6, 10**6, 10**6, 10**6), False)
|
||||||
|
|
||||||
# Wait for l1 notice
|
# Wait for l1 notice
|
||||||
l1.daemon.wait_for_log(r'Peer transient failure in CHANNELD_NORMAL: channeld WARNING: .*: update_fee \d+ outside range 1875-75000')
|
l1.daemon.wait_for_log(r'Peer transient failure in CHANNELD_NORMAL: channeld WARNING: .*: update_fee \d+ outside range 1875-75000')
|
||||||
|
Loading…
Reference in New Issue
Block a user