URI: 
       tln: shortcut some OpenChannel fields to traversing too much - electrum - Electrum Bitcoin wallet
  HTML git clone https://git.parazyd.org/electrum
   DIR Log
   DIR Files
   DIR Refs
   DIR Submodules
       ---
   DIR commit 42a56df99633e714952a2bff353d06b1a4c3f3d9
   DIR parent 77e9abc6552a71e2382c72a8a97aab782ba7e513
  HTML Author: Janus <ysangkok@gmail.com>
       Date:   Wed, 27 Jun 2018 18:49:51 +0200
       
       ln: shortcut some OpenChannel fields to traversing too much
       
       Diffstat:
         M gui/qt/channels_list.py             |       4 ++--
         M lib/lnbase.py                       |     102 ++++++++++++++++----------------
         M lib/lnhtlc.py                       |      19 +++++++++++++++++++
         M lib/lnworker.py                     |      18 +++++++++---------
       
       4 files changed, 81 insertions(+), 62 deletions(-)
       ---
   DIR diff --git a/gui/qt/channels_list.py b/gui/qt/channels_list.py
       t@@ -50,8 +50,8 @@ class ChannelsList(MyTreeWidget):
            def do_update_rows(self):
                self.clear()
                for chan in self.parent.wallet.lnworker.channels.values():
       -            item = SortableTreeWidgetItem(self.format_fields(chan.state))
       -            item.setData(0, QtCore.Qt.UserRole, chan.state.channel_id)
       +            item = SortableTreeWidgetItem(self.format_fields(chan))
       +            item.setData(0, QtCore.Qt.UserRole, chan.channel_id)
                    self.insertTopLevelItem(0, item)
        
            def get_toolbar(self):
   DIR diff --git a/lib/lnbase.py b/lib/lnbase.py
       t@@ -776,7 +776,7 @@ class Peer(PrintError):
            def on_announcement_signatures(self, payload):
                channel_id = payload['channel_id']
                chan = self.channels[payload['channel_id']]
       -        if chan.state.local_state.was_announced:
       +        if chan.local_state.was_announced:
                    h, local_node_sig, local_bitcoin_sig = self.send_announcement_signatures(chan)
                else:
                    self.announcement_signatures[channel_id].put_nowait(payload)
       t@@ -933,17 +933,17 @@ class Peer(PrintError):
            @aiosafe
            async def reestablish_channel(self, chan):
                await self.initialized
       -        chan_id = chan.state.channel_id
       +        chan_id = chan.channel_id
                self.channel_state[chan_id] = 'REESTABLISHING'
                self.network.trigger_callback('channel', chan)
                self.send_message(gen_msg("channel_reestablish",
                    channel_id=chan_id,
       -            next_local_commitment_number=chan.state.local_state.ctn+1,
       -            next_remote_revocation_number=chan.state.remote_state.ctn
       +            next_local_commitment_number=chan.local_state.ctn+1,
       +            next_remote_revocation_number=chan.remote_state.ctn
                ))
                await self.channel_reestablished[chan_id]
                self.channel_state[chan_id] = 'OPENING'
       -        if chan.state.local_state.funding_locked_received and chan.state.short_channel_id:
       +        if chan.local_state.funding_locked_received and chan.state.short_channel_id:
                    self.mark_open(chan)
                self.network.trigger_callback('channel', chan)
        
       t@@ -956,26 +956,26 @@ class Peer(PrintError):
                    return
                channel_reestablish_msg = payload
                remote_ctn = int.from_bytes(channel_reestablish_msg["next_local_commitment_number"], 'big')
       -        if remote_ctn != chan.state.remote_state.ctn + 1:
       -            raise Exception("expected remote ctn {}, got {}".format(chan.state.remote_state.ctn + 1, remote_ctn))
       +        if remote_ctn != chan.remote_state.ctn + 1:
       +            raise Exception("expected remote ctn {}, got {}".format(chan.remote_state.ctn + 1, remote_ctn))
                local_ctn = int.from_bytes(channel_reestablish_msg["next_remote_revocation_number"], 'big')
       -        if local_ctn != chan.state.local_state.ctn:
       -            raise Exception("expected local ctn {}, got {}".format(chan.state.local_state.ctn, local_ctn))
       +        if local_ctn != chan.local_state.ctn:
       +            raise Exception("expected local ctn {}, got {}".format(chan.local_state.ctn, local_ctn))
                their = channel_reestablish_msg["my_current_per_commitment_point"]
       -        our = chan.state.remote_state.current_per_commitment_point
       +        our = chan.remote_state.current_per_commitment_point
                if our is None:
       -            our = chan.state.remote_state.next_per_commitment_point
       +            our = chan.remote_state.next_per_commitment_point
                if our != their:
                    raise Exception("Remote PCP mismatch: {} {}".format(bh2u(our), bh2u(their)))
                self.channel_reestablished[chan_id].set_result(True)
        
            def funding_locked(self, chan):
       -        channel_id = chan.state.channel_id
       +        channel_id = chan.channel_id
                per_commitment_secret_index = 2**48 - 2
                per_commitment_point_second = secret_to_pubkey(int.from_bytes(
       -            get_per_commitment_secret_from_seed(chan.state.local_state.per_commitment_secret_seed, per_commitment_secret_index), 'big'))
       +            get_per_commitment_secret_from_seed(chan.local_state.per_commitment_secret_seed, per_commitment_secret_index), 'big'))
                self.send_message(gen_msg("funding_locked", channel_id=channel_id, next_per_commitment_point=per_commitment_point_second))
       -        if chan.state.local_state.funding_locked_received:
       +        if chan.local_state.funding_locked_received:
                    self.mark_open(chan)
        
            def on_funding_locked(self, payload):
       t@@ -983,11 +983,11 @@ class Peer(PrintError):
                chan = self.channels.get(channel_id)
                if not chan:
                    raise Exception("Got unknown funding_locked", channel_id)
       -        if not chan.state.local_state.funding_locked_received:
       -            our_next_point = chan.state.remote_state.next_per_commitment_point
       +        if not chan.local_state.funding_locked_received:
       +            our_next_point = chan.remote_state.next_per_commitment_point
                    their_next_point = payload["next_per_commitment_point"]
       -            new_remote_state = chan.state.remote_state._replace(next_per_commitment_point=their_next_point, current_per_commitment_point=our_next_point)
       -            new_local_state = chan.state.local_state._replace(funding_locked_received = True)
       +            new_remote_state = chan.remote_state._replace(next_per_commitment_point=their_next_point, current_per_commitment_point=our_next_point)
       +            new_local_state = chan.local_state._replace(funding_locked_received = True)
                    chan.state = chan.state._replace(remote_state=new_remote_state, local_state=new_local_state)
                    self.lnworker.save_channel(chan)
                if chan.state.short_channel_id:
       t@@ -999,8 +999,8 @@ class Peer(PrintError):
        
                Runs on the Network thread.
                """
       -        if not chan.state.local_state.was_announced and funding_tx_depth >= 6:
       -            chan.state = chan.state._replace(local_state=chan.state.local_state._replace(was_announced=True))
       +        if not chan.local_state.was_announced and funding_tx_depth >= 6:
       +            chan.state = chan.state._replace(local_state=chan.local_state._replace(was_announced=True))
                    coro = self.handle_announcements(chan)
                    self.lnworker.save_channel(chan)
                    asyncio.run_coroutine_threadsafe(coro, self.network.asyncio_loop)
       t@@ -1008,10 +1008,10 @@ class Peer(PrintError):
            @aiosafe
            async def handle_announcements(self, chan):
                h, local_node_sig, local_bitcoin_sig = self.send_announcement_signatures(chan)
       -        announcement_signatures_msg = await self.announcement_signatures[chan.state.channel_id].get()
       +        announcement_signatures_msg = await self.announcement_signatures[chan.channel_id].get()
                remote_node_sig = announcement_signatures_msg["node_signature"]
                remote_bitcoin_sig = announcement_signatures_msg["bitcoin_signature"]
       -        if not ecc.verify_signature(chan.state.remote_config.multisig_key.pubkey, remote_bitcoin_sig, h):
       +        if not ecc.verify_signature(chan.remote_config.multisig_key.pubkey, remote_bitcoin_sig, h):
                    raise Exception("bitcoin_sig invalid in announcement_signatures")
                if not ecc.verify_signature(self.pubkey, remote_node_sig, h):
                    raise Exception("node_sig invalid in announcement_signatures")
       t@@ -1019,7 +1019,7 @@ class Peer(PrintError):
                node_sigs = [local_node_sig, remote_node_sig]
                bitcoin_sigs = [local_bitcoin_sig, remote_bitcoin_sig]
                node_ids = [privkey_to_pubkey(self.privkey), self.pubkey]
       -        bitcoin_keys = [chan.state.local_config.multisig_key.pubkey, chan.state.remote_config.multisig_key.pubkey]
       +        bitcoin_keys = [chan.local_config.multisig_key.pubkey, chan.remote_config.multisig_key.pubkey]
        
                if node_ids[0] > node_ids[1]:
                    node_sigs.reverse()
       t@@ -1047,10 +1047,10 @@ class Peer(PrintError):
                print("SENT CHANNEL ANNOUNCEMENT")
        
            def mark_open(self, chan):
       -        if self.channel_state[chan.state.channel_id] == "OPEN":
       +        if self.channel_state[chan.channel_id] == "OPEN":
                    return
       -        assert chan.state.local_state.funding_locked_received
       -        self.channel_state[chan.state.channel_id] = "OPEN"
       +        assert chan.local_state.funding_locked_received
       +        self.channel_state[chan.channel_id] = "OPEN"
                self.network.trigger_callback('channel', chan.state)
                # add channel to database
                sorted_keys = list(sorted([self.pubkey, self.lnworker.pubkey]))
       t@@ -1062,8 +1062,8 @@ class Peer(PrintError):
        
            def send_announcement_signatures(self, chan):
        
       -        bitcoin_keys = [chan.state.local_config.multisig_key.pubkey,
       -                        chan.state.remote_config.multisig_key.pubkey]
       +        bitcoin_keys = [chan.local_config.multisig_key.pubkey,
       +                        chan.remote_config.multisig_key.pubkey]
        
                node_ids = [privkey_to_pubkey(self.privkey),
                            self.pubkey]
       t@@ -1085,10 +1085,10 @@ class Peer(PrintError):
                )
                to_hash = chan_ann[256+2:]
                h = bitcoin.Hash(to_hash)
       -        bitcoin_signature = ecc.ECPrivkey(chan.state.local_config.multisig_key.privkey).sign(h, sigencode_string_canonize, sigdecode_string)
       +        bitcoin_signature = ecc.ECPrivkey(chan.local_config.multisig_key.privkey).sign(h, sigencode_string_canonize, sigdecode_string)
                node_signature = ecc.ECPrivkey(self.privkey).sign(h, sigencode_string_canonize, sigdecode_string)
                self.send_message(gen_msg("announcement_signatures",
       -            channel_id=chan.state.channel_id,
       +            channel_id=chan.channel_id,
                    short_channel_id=chan.state.short_channel_id,
                    node_signature=node_signature,
                    bitcoin_signature=bitcoin_signature
       t@@ -1124,7 +1124,7 @@ class Peer(PrintError):
        
            @aiosafe
            async def pay(self, path, chan, amount_msat, payment_hash, pubkey_in_invoice, min_final_cltv_expiry):
       -        assert self.channel_state[chan.state.channel_id] == "OPEN"
       +        assert self.channel_state[chan.channel_id] == "OPEN"
                assert amount_msat > 0, "amount_msat is not greater zero"
                height = self.network.get_local_height()
                route = self.network.path_finder.create_route_from_path(path, self.lnworker.pubkey)
       t@@ -1140,38 +1140,38 @@ class Peer(PrintError):
                self.secret_key = os.urandom(32)
                hops_data += [OnionHopsDataSingle(OnionPerHop(b"\x00"*8, amount_msat.to_bytes(8, "big"), (final_cltv_expiry_without_deltas).to_bytes(4, "big")))]
                onion = new_onion_packet([x.node_id for x in route], self.secret_key, hops_data, associated_data)
       -        msat_local = chan.state.local_state.amount_msat - (amount_msat + total_fee)
       -        msat_remote = chan.state.remote_state.amount_msat + (amount_msat + total_fee)
       +        msat_local = chan.local_state.amount_msat - (amount_msat + total_fee)
       +        msat_remote = chan.remote_state.amount_msat + (amount_msat + total_fee)
                htlc = UpdateAddHtlc(amount_msat, payment_hash, final_cltv_expiry_with_deltas, total_fee)
                amount_msat += total_fee
        
       -        self.send_message(gen_msg("update_add_htlc", channel_id=chan.state.channel_id, id=chan.state.local_state.next_htlc_id, cltv_expiry=final_cltv_expiry_with_deltas, amount_msat=amount_msat, payment_hash=payment_hash, onion_routing_packet=onion.to_bytes()))
       +        self.send_message(gen_msg("update_add_htlc", channel_id=chan.channel_id, id=chan.local_state.next_htlc_id, cltv_expiry=final_cltv_expiry_with_deltas, amount_msat=amount_msat, payment_hash=payment_hash, onion_routing_packet=onion.to_bytes()))
        
                chan.add_htlc(htlc)
       -        self.attempted_route[(chan.state.channel_id, htlc.htlc_id)] = route
       +        self.attempted_route[(chan.channel_id, htlc.htlc_id)] = route
        
                sig_64, htlc_sigs = chan.sign_next_commitment()
                htlc_sig = htlc_sigs[0]
        
       -        self.send_message(gen_msg("commitment_signed", channel_id=chan.state.channel_id, signature=sig_64, num_htlcs=1, htlc_signature=htlc_sig))
       +        self.send_message(gen_msg("commitment_signed", channel_id=chan.channel_id, signature=sig_64, num_htlcs=1, htlc_signature=htlc_sig))
        
                await self.receive_revoke(chan)
        
                self.revoke(chan)
        
       -        fulfill_coro = asyncio.ensure_future(self.update_fulfill_htlc[chan.state.channel_id].get())
       -        failure_coro = asyncio.ensure_future(self.update_fail_htlc[chan.state.channel_id].get())
       +        fulfill_coro = asyncio.ensure_future(self.update_fulfill_htlc[chan.channel_id].get())
       +        failure_coro = asyncio.ensure_future(self.update_fail_htlc[chan.channel_id].get())
        
                done, pending = await asyncio.wait([fulfill_coro, failure_coro], return_when=FIRST_COMPLETED)
                if failure_coro.done():
                    sig_64, htlc_sigs = chan.sign_next_commitment()
       -            self.send_message(gen_msg("commitment_signed", channel_id=chan.state.channel_id, signature=sig_64, num_htlcs=1, htlc_signature=htlc_sigs[0]))
       -            while (await self.commitment_signed[chan.state.channel_id].get())["htlc_signature"] != b"":
       +            self.send_message(gen_msg("commitment_signed", channel_id=chan.channel_id, signature=sig_64, num_htlcs=1, htlc_signature=htlc_sigs[0]))
       +            while (await self.commitment_signed[chan.channel_id].get())["htlc_signature"] != b"":
                        self.revoke(chan)
                    await self.receive_revoke(chan)
                    chan.fail_htlc(htlc)
                    sig_64, htlc_sigs = chan.sign_next_commitment()
       -            self.send_message(gen_msg("commitment_signed", channel_id=chan.state.channel_id, signature=sig_64, num_htlcs=0))
       +            self.send_message(gen_msg("commitment_signed", channel_id=chan.channel_id, signature=sig_64, num_htlcs=0))
                    await self.receive_revoke(chan)
                    fulfill_coro.cancel()
                    self.lnworker.save_channel(chan)
       t@@ -1182,33 +1182,33 @@ class Peer(PrintError):
        
                chan.receive_htlc_settle(update_fulfill_htlc_msg["payment_preimage"], int.from_bytes(update_fulfill_htlc_msg["id"], "big"))
        
       -        while (await self.commitment_signed[chan.state.channel_id].get())["htlc_signature"] != b"":
       +        while (await self.commitment_signed[chan.channel_id].get())["htlc_signature"] != b"":
                    self.revoke(chan)
                # TODO process above commitment transactions
        
       -        bare_ctx = make_commitment_using_open_channel(chan.state, chan.state.remote_state.ctn + 1, False, chan.state.remote_state.next_per_commitment_point,
       +        bare_ctx = make_commitment_using_open_channel(chan.state, chan.remote_state.ctn + 1, False, chan.remote_state.next_per_commitment_point,
                    msat_remote, msat_local)
        
       -        sig_64 = sign_and_get_sig_string(bare_ctx, chan.state.local_config, chan.state.remote_config)
       -        self.send_message(gen_msg("commitment_signed", channel_id=chan.state.channel_id, signature=sig_64, num_htlcs=0))
       +        sig_64 = sign_and_get_sig_string(bare_ctx, chan.local_config, chan.remote_config)
       +        self.send_message(gen_msg("commitment_signed", channel_id=chan.channel_id, signature=sig_64, num_htlcs=0))
        
                await self.receive_revoke(chan)
        
                self.lnworker.save_channel(chan)
        
            async def receive_revoke(self, m):
       -        revoke_and_ack_msg = await self.revoke_and_ack[m.state.channel_id].get()
       +        revoke_and_ack_msg = await self.revoke_and_ack[m.channel_id].get()
                m.receive_revocation(RevokeAndAck(revoke_and_ack_msg["per_commitment_secret"], revoke_and_ack_msg["next_per_commitment_point"]))
        
            def revoke(self, m):
                rev, _ = m.revoke_current_commitment()
                self.send_message(gen_msg("revoke_and_ack",
       -            channel_id=m.state.channel_id,
       +            channel_id=m.channel_id,
                    per_commitment_secret=rev.per_commitment_secret,
                    next_per_commitment_point=rev.next_per_commitment_point))
        
            async def receive_commitment(self, m):
       -        commitment_signed_msg = await self.commitment_signed[m.state.channel_id].get()
       +        commitment_signed_msg = await self.commitment_signed[m.channel_id].get()
                data = commitment_signed_msg["htlc_signature"]
                htlc_sigs = [data[i:i+64] for i in range(0, len(data), 64)]
                m.receive_new_commitment(commitment_signed_msg["signature"], htlc_sigs)
       t@@ -1217,10 +1217,10 @@ class Peer(PrintError):
            @aiosafe
            async def receive_commitment_revoke_ack(self, htlc, decoded, payment_preimage):
                chan = self.channels[htlc['channel_id']]
       -        channel_id = chan.state.channel_id
       +        channel_id = chan.channel_id
                expected_received_msat = int(decoded.amount * COIN * 1000)
                htlc_id = int.from_bytes(htlc["id"], 'big')
       -        assert htlc_id == chan.state.remote_state.next_htlc_id, (htlc_id, chan.state.remote_state.next_htlc_id)
       +        assert htlc_id == chan.remote_state.next_htlc_id, (htlc_id, chan.remote_state.next_htlc_id)
        
                assert self.channel_state[channel_id] == "OPEN"
        
       t@@ -1265,7 +1265,7 @@ class Peer(PrintError):
                self.print_error("commitment_signed", payload)
                channel_id = payload['channel_id']
                chan = self.channels[channel_id]
       -        chan.state = chan.state._replace(local_state=chan.state.local_state._replace(current_commitment_signature=payload['signature']))
       +        chan.state = chan.state._replace(local_state=chan.local_state._replace(current_commitment_signature=payload['signature']))
                self.lnworker.save_channel(chan)
                self.commitment_signed[channel_id].put_nowait(payload)
        
   DIR diff --git a/lib/lnhtlc.py b/lib/lnhtlc.py
       t@@ -448,3 +448,22 @@ class HTLCStateMachine(PrintError):
            def receive_update_fee(self, fee):
                self.pending_feerate = fee
        
       +    @property
       +    def local_state(self):
       +        return self.state.local_state
       +
       +    @property
       +    def remote_state(self):
       +        return self.state.remote_state
       +
       +    @property
       +    def remote_config(self):
       +        return self.state.remote_config
       +
       +    @property
       +    def local_config(self):
       +        return self.state.local_config
       +
       +    @property
       +    def channel_id(self):
       +        return self.state.channel_id
   DIR diff --git a/lib/lnworker.py b/lib/lnworker.py
       t@@ -94,7 +94,7 @@ class LNWorker(PrintError):
                self.channels = {x.channel_id: HTLCStateMachine(x) for x in map(reconstruct_namedtuples, wallet.storage.get("channels", []))}
                self.invoices = wallet.storage.get('lightning_invoices', {})
                peer_list = network.config.get('lightning_peers', node_list)
       -        self.channel_state = {chan.state.channel_id: "DISCONNECTED" for chan in self.channels.values()}
       +        self.channel_state = {chan.channel_id: "DISCONNECTED" for chan in self.channels.values()}
                for chan_id, chan in self.channels.items():
                    self.network.lnwatcher.watch_channel(chan, self.on_channel_utxos)
                for host, port, pubkey in peer_list:
       t@@ -115,9 +115,9 @@ class LNWorker(PrintError):
        
            def save_channel(self, openchannel):
                assert type(openchannel) is HTLCStateMachine
       -        if openchannel.state.channel_id not in self.channel_state:
       -            self.channel_state[openchannel.state.channel_id] = "OPENING"
       -        self.channels[openchannel.state.channel_id] = openchannel
       +        if openchannel.channel_id not in self.channel_state:
       +            self.channel_state[openchannel.channel_id] = "OPENING"
       +        self.channels[openchannel.channel_id] = openchannel
                for node_id, peer in self.peers.items():
                    peer.channels = self.channels_for_peer(node_id)
                if openchannel.state.remote_state.next_per_commitment_point == openchannel.state.remote_state.current_per_commitment_point:
       t@@ -133,7 +133,7 @@ class LNWorker(PrintError):
        
                If the Funding TX has not been mined, return None
                """
       -        assert self.channel_state[chan.state.channel_id] in ["OPEN", "OPENING"]
       +        assert self.channel_state[chan.channel_id] in ["OPEN", "OPENING"]
                peer = self.peers[chan.state.node_id]
                conf = self.wallet.get_tx_height(chan.state.funding_outpoint.txid)[1]
                if conf >= chan.state.constraints.funding_txn_minimum_depth:
       t@@ -150,8 +150,8 @@ class LNWorker(PrintError):
            def on_channel_utxos(self, chan, utxos):
                outpoints = [Outpoint(x["tx_hash"], x["tx_pos"]) for x in utxos]
                if chan.state.funding_outpoint not in outpoints:
       -            self.channel_state[chan.state.channel_id] = "CLOSED"
       -        elif self.channel_state[chan.state.channel_id] == 'DISCONNECTED':
       +            self.channel_state[chan.channel_id] = "CLOSED"
       +        elif self.channel_state[chan.channel_id] == 'DISCONNECTED':
                    peer = self.peers[chan.state.node_id]
                    coro = peer.reestablish_channel(chan)
                    asyncio.run_coroutine_threadsafe(coro, self.network.asyncio_loop)
       t@@ -159,14 +159,14 @@ class LNWorker(PrintError):
            def on_network_update(self, event, *args):
                for chan in self.channels.values():
                    peer = self.peers[chan.state.node_id]
       -            if self.channel_state[chan.state.channel_id] == "OPENING":
       +            if self.channel_state[chan.channel_id] == "OPENING":
                        res = self.save_short_chan_id(chan)
                        if not res:
                            self.print_error("network update but funding tx is still not at sufficient depth")
                            continue
                        # this results in the channel being marked OPEN
                        peer.funding_locked(chan)
       -            elif self.channel_state[chan.state.channel_id] == "OPEN":
       +            elif self.channel_state[chan.channel_id] == "OPEN":
                        conf = self.wallet.get_tx_height(chan.state.funding_outpoint.txid)[1]
                        peer.on_network_update(chan, conf)