|
@ -63,7 +63,7 @@ class FeeUpdate: |
|
|
return self.rate |
|
|
return self.rate |
|
|
# implicit return None |
|
|
# implicit return None |
|
|
|
|
|
|
|
|
class UpdateAddHtlc(namedtuple('UpdateAddHtlc', ['amount_msat', 'payment_hash', 'cltv_expiry', 'settled', 'locked_in', 'htlc_id'])): |
|
|
class UpdateAddHtlc(namedtuple('UpdateAddHtlc', ['amount_msat', 'payment_hash', 'cltv_expiry', 'locked_in', 'htlc_id'])): |
|
|
__slots__ = () |
|
|
__slots__ = () |
|
|
def __new__(cls, *args, **kwargs): |
|
|
def __new__(cls, *args, **kwargs): |
|
|
if len(args) > 0: |
|
|
if len(args) > 0: |
|
@ -71,7 +71,6 @@ class UpdateAddHtlc(namedtuple('UpdateAddHtlc', ['amount_msat', 'payment_hash', |
|
|
if type(args[1]) is str: |
|
|
if type(args[1]) is str: |
|
|
args[1] = bfh(args[1]) |
|
|
args[1] = bfh(args[1]) |
|
|
args[3] = {HTLCOwner(int(x)): y for x,y in args[3].items()} |
|
|
args[3] = {HTLCOwner(int(x)): y for x,y in args[3].items()} |
|
|
args[4] = {HTLCOwner(int(x)): y for x,y in args[4].items()} |
|
|
|
|
|
return super().__new__(cls, *args) |
|
|
return super().__new__(cls, *args) |
|
|
if type(kwargs['payment_hash']) is str: |
|
|
if type(kwargs['payment_hash']) is str: |
|
|
kwargs['payment_hash'] = bfh(kwargs['payment_hash']) |
|
|
kwargs['payment_hash'] = bfh(kwargs['payment_hash']) |
|
@ -79,10 +78,6 @@ class UpdateAddHtlc(namedtuple('UpdateAddHtlc', ['amount_msat', 'payment_hash', |
|
|
kwargs['locked_in'] = {LOCAL: None, REMOTE: None} |
|
|
kwargs['locked_in'] = {LOCAL: None, REMOTE: None} |
|
|
else: |
|
|
else: |
|
|
kwargs['locked_in'] = {HTLCOwner(int(x)): y for x,y in kwargs['locked_in']} |
|
|
kwargs['locked_in'] = {HTLCOwner(int(x)): y for x,y in kwargs['locked_in']} |
|
|
if 'settled' not in kwargs: |
|
|
|
|
|
kwargs['settled'] = {LOCAL: None, REMOTE: None} |
|
|
|
|
|
else: |
|
|
|
|
|
kwargs['settled'] = {HTLCOwner(int(x)): y for x,y in kwargs['settled']} |
|
|
|
|
|
return super().__new__(cls, **kwargs) |
|
|
return super().__new__(cls, **kwargs) |
|
|
|
|
|
|
|
|
is_key = lambda k: k.endswith("_basepoint") or k.endswith("_key") |
|
|
is_key = lambda k: k.endswith("_basepoint") or k.endswith("_key") |
|
@ -176,6 +171,8 @@ class HTLCStateMachine(PrintError): |
|
|
|
|
|
|
|
|
self.lnwatcher = None |
|
|
self.lnwatcher = None |
|
|
|
|
|
|
|
|
|
|
|
self.settled = {LOCAL: state.get('settled_local', []), REMOTE: state.get('settled_remote', [])} |
|
|
|
|
|
|
|
|
def set_state(self, state: str): |
|
|
def set_state(self, state: str): |
|
|
self._state = state |
|
|
self._state = state |
|
|
|
|
|
|
|
@ -429,10 +426,14 @@ class HTLCStateMachine(PrintError): |
|
|
""" |
|
|
""" |
|
|
old_amount = self.htlcsum(self.gen_htlc_indices(subject, False)) |
|
|
old_amount = self.htlcsum(self.gen_htlc_indices(subject, False)) |
|
|
|
|
|
|
|
|
|
|
|
removed = [] |
|
|
for x in self.log[-subject]: |
|
|
for x in self.log[-subject]: |
|
|
if type(x) is not SettleHtlc: continue |
|
|
if type(x) is not SettleHtlc: continue |
|
|
htlc = self.lookup_htlc(self.log[subject], x.htlc_id) |
|
|
htlc = self.lookup_htlc(self.log[subject], x.htlc_id) |
|
|
htlc.settled[subject] = self.current_height[subject] |
|
|
self.settled[subject].append(htlc.amount_msat) |
|
|
|
|
|
self.log[subject].remove(htlc) |
|
|
|
|
|
removed.append(x) |
|
|
|
|
|
for x in removed: self.log[-subject].remove(x) |
|
|
|
|
|
|
|
|
return old_amount - self.htlcsum(self.gen_htlc_indices(subject, False)) |
|
|
return old_amount - self.htlcsum(self.gen_htlc_indices(subject, False)) |
|
|
|
|
|
|
|
@ -465,15 +466,8 @@ class HTLCStateMachine(PrintError): |
|
|
def balance(self, subject): |
|
|
def balance(self, subject): |
|
|
initial = self.local_config.initial_msat if subject == LOCAL else self.remote_config.initial_msat |
|
|
initial = self.local_config.initial_msat if subject == LOCAL else self.remote_config.initial_msat |
|
|
|
|
|
|
|
|
for direction in (SENT, RECEIVED): |
|
|
initial -= sum(self.settled[subject]) |
|
|
for x in self.log[-direction]: |
|
|
initial += sum(self.settled[-subject]) |
|
|
if type(x) is not SettleHtlc: continue |
|
|
|
|
|
htlc = self.lookup_htlc(self.log[direction], x.htlc_id) |
|
|
|
|
|
htlc_height = htlc.settled[direction] |
|
|
|
|
|
if htlc_height is not None and htlc_height <= self.current_height[direction]: |
|
|
|
|
|
# so we will subtract when direction == subject. |
|
|
|
|
|
# example subject=LOCAL, direction=SENT: we subtract |
|
|
|
|
|
initial -= htlc.amount_msat * subject * direction |
|
|
|
|
|
|
|
|
|
|
|
assert initial == (self.local_state.amount_msat if subject == LOCAL else self.remote_state.amount_msat) |
|
|
assert initial == (self.local_state.amount_msat if subject == LOCAL else self.remote_state.amount_msat) |
|
|
return initial |
|
|
return initial |
|
@ -528,11 +522,10 @@ class HTLCStateMachine(PrintError): |
|
|
_, this_point, _ = self.points |
|
|
_, this_point, _ = self.points |
|
|
return self.make_commitment(LOCAL, this_point) |
|
|
return self.make_commitment(LOCAL, this_point) |
|
|
|
|
|
|
|
|
@property |
|
|
def total_msat(self, sub): |
|
|
def total_msat(self): |
|
|
return sum(self.settled[sub]) |
|
|
return {LOCAL: self.htlcsum(self.gen_htlc_indices(LOCAL, False, True)), REMOTE: self.htlcsum(self.gen_htlc_indices(REMOTE, False, True))} |
|
|
|
|
|
|
|
|
|
|
|
def gen_htlc_indices(self, subject, only_pending, include_settled=False): |
|
|
def gen_htlc_indices(self, subject, only_pending): |
|
|
""" |
|
|
""" |
|
|
only_pending: require the htlc's settlement to be pending (needs additional signatures/acks) |
|
|
only_pending: require the htlc's settlement to be pending (needs additional signatures/acks) |
|
|
include_settled: include settled (totally done with) htlcs |
|
|
include_settled: include settled (totally done with) htlcs |
|
@ -543,17 +536,10 @@ class HTLCStateMachine(PrintError): |
|
|
for htlc in update_log: |
|
|
for htlc in update_log: |
|
|
if type(htlc) is not UpdateAddHtlc: |
|
|
if type(htlc) is not UpdateAddHtlc: |
|
|
continue |
|
|
continue |
|
|
height = self.current_height[-subject] |
|
|
|
|
|
locked_in = htlc.locked_in[subject] |
|
|
locked_in = htlc.locked_in[subject] |
|
|
|
|
|
|
|
|
if locked_in is None or only_pending == (SettleHtlc(htlc.htlc_id) in other_log): |
|
|
if locked_in is None or only_pending == (SettleHtlc(htlc.htlc_id) in other_log): |
|
|
continue |
|
|
continue |
|
|
|
|
|
|
|
|
settled_cutoff = self.local_state.ctn if subject == LOCAL else self.remote_state.ctn |
|
|
|
|
|
|
|
|
|
|
|
if not include_settled and htlc.settled[subject] is not None and settled_cutoff >= htlc.settled[subject]: |
|
|
|
|
|
continue |
|
|
|
|
|
|
|
|
|
|
|
res.append(htlc) |
|
|
res.append(htlc) |
|
|
return res |
|
|
return res |
|
|
|
|
|
|
|
@ -651,6 +637,8 @@ class HTLCStateMachine(PrintError): |
|
|
"remote_log": [(type(x).__name__, x) for x in remote_filtered], |
|
|
"remote_log": [(type(x).__name__, x) for x in remote_filtered], |
|
|
"local_log": [(type(x).__name__, x) for x in local_filtered], |
|
|
"local_log": [(type(x).__name__, x) for x in local_filtered], |
|
|
"onion_keys": {str(k): bh2u(v) for k, v in self.onion_keys.items()}, |
|
|
"onion_keys": {str(k): bh2u(v) for k, v in self.onion_keys.items()}, |
|
|
|
|
|
"settled_local": self.settled[LOCAL], |
|
|
|
|
|
"settled_remote": self.settled[REMOTE], |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
# htlcs number must be monotonically increasing, |
|
|
# htlcs number must be monotonically increasing, |
|
|