Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

(WIP) AuxPoW: Miscellaneous gettransaction Tor improvements #183

Open
wants to merge 20 commits into
base: auxpow
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
20 commits
Select commit Hold shift + click to select a range
59670d7
Use cp_height for checkpoints.
JeremyRand Nov 5, 2019
d76b8e7
Add test for blockchain.get_chainwork
JeremyRand Nov 5, 2019
8882423
Add timestamp and chainwork to checkpoint
JeremyRand Nov 5, 2019
ceacede
Eliminate obsolete special case from request_chunk
JeremyRand Jun 7, 2019
97fd7d2
Test get_chainwork with a variety of checkpoint heights
JeremyRand Nov 5, 2019
f98f365
Add bits to checkpoint
JeremyRand Nov 5, 2019
9246429
Refactor cp_height checkpoints' usage of header parsing
JeremyRand Aug 24, 2019
5f7fd05
Add prev_hash to cp_height checkpoint logic
JeremyRand Jun 7, 2019
aef4d8f
Load cp_height checkpoints from JSON file instead of constants.py
JeremyRand Nov 5, 2019
629b1b6
Use verifier.py's hash_merkle_root for checkpoints
JeremyRand Aug 24, 2019
ea82537
(WIP) Namecoin / AuxPoW: Adapt cp_height checkpoints for AuxPoW
JeremyRand Aug 24, 2019
41bda45
(WIP) Namecoin / AuxPoW: Fix get_block_header parsing for cp_height
JeremyRand Sep 26, 2019
cf81d2b
(WIP) Namecoin / AuxPoW: Fix checkpoint exporting
JeremyRand Sep 26, 2019
7416f93
verifier: Get single header when cheaper than chunk
JeremyRand Sep 23, 2019
07c798d
verifier: Support running without a wallet
JeremyRand Sep 26, 2019
941766e
commands: enable optional verification in gettransaction
JeremyRand Sep 26, 2019
f1dd790
(WIP) verifier: Support stream isolation
JeremyRand Nov 5, 2019
5f5d2ae
(WIP) commands: Support verifier stream isolation in gettransaction
JeremyRand Nov 5, 2019
09f843a
(WIP) commands: Make gettransaction use cp_height header validation
JeremyRand Nov 5, 2019
a7727ee
(WIP) verifier: Read headers more quickly
JeremyRand Nov 5, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
221 changes: 152 additions & 69 deletions electrum/blockchain.py

Large diffs are not rendered by default.

1,178 changes: 8 additions & 1,170 deletions electrum/checkpoints.json

Large diffs are not rendered by default.

3,130 changes: 8 additions & 3,122 deletions electrum/checkpoints_testnet.json

Large diffs are not rendered by default.

19 changes: 17 additions & 2 deletions electrum/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@
from .synchronizer import Notifier
from .wallet import Abstract_Wallet, create_new_wallet, restore_wallet_from_text
from .address_synchronizer import TX_HEIGHT_LOCAL
from .verifier import SPV
from .mnemonic import Mnemonic
from .lnutil import SENT, RECEIVED
from .lnpeer import channel_id_from_funding_tx
Expand Down Expand Up @@ -679,17 +680,29 @@ async def listaddresses(self, receiving=False, change=False, labels=False, froze
return out

@command('n')
async def gettransaction(self, txid, wallet: Abstract_Wallet = None):
async def gettransaction(self, txid, verify=False, height=None, wallet: Abstract_Wallet = None):
"""Retrieve a transaction. """
if verify:
if height is None:
raise Exception("Missing height")
verifier = SPV(self.network, None)._request_and_verify_single_proof(txid, height, use_individual_header_proof=(height < constants.net.max_checkpoint()), stream_id=stream_id)
tx = None
if wallet:
tx = wallet.db.get_transaction(txid)
if tx is None:
raw = await self.network.get_transaction(txid)
raw_getter = self.network.get_transaction(txid)
if verify:
_, raw = await asyncio.gather(verifier, raw_getter)
else:
raw = await raw_getter
if raw:
tx = Transaction(raw)
else:
raise Exception("Unknown transaction")
elif verify:
await verifier
if verify and tx.txid() != txid:
raise Exception("Wrong txid")
return tx.as_dict()

@command('')
Expand Down Expand Up @@ -1020,6 +1033,8 @@ def eval_bool(x: str) -> bool:
'fee_level': (None, "Float between 0.0 and 1.0, representing fee slider position"),
'from_height': (None, "Only show transactions that confirmed after given block height"),
'to_height': (None, "Only show transactions that confirmed before given block height"),
'verify': (None, "Verify transaction via SPV"),
'height': (None, "Block height"),
}


Expand Down
13 changes: 8 additions & 5 deletions electrum/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ class AbstractNet:

@classmethod
def max_checkpoint(cls) -> int:
return max(0, len(cls.CHECKPOINTS) * 2016 - 1)
return cls.CHECKPOINTS['height']

@classmethod
def rev_genesis_bytes(cls) -> bytes:
Expand All @@ -67,7 +67,10 @@ class BitcoinMainnet(AbstractNet):
GENESIS = "000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f"
DEFAULT_PORTS = {'t': '50001', 's': '50002'}
DEFAULT_SERVERS = read_json('servers.json', {})
CHECKPOINTS = read_json('checkpoints.json', [])
# To generate this JSON file, connect to a trusted server, and then run
# this from the console:
# network.run_from_another_thread(network.interface.export_purported_checkpoints(height, path))
CHECKPOINTS = read_json('checkpoints.json', {'height': 0})
BLOCK_HEIGHT_FIRST_LIGHTNING_CHANNELS = 497000

XPRV_HEADERS = {
Expand Down Expand Up @@ -104,7 +107,7 @@ class BitcoinTestnet(AbstractNet):
GENESIS = "000000000933ea01ad0ee984209779baaec3ced90fa3f408719526f8d77f4943"
DEFAULT_PORTS = {'t': '51001', 's': '51002'}
DEFAULT_SERVERS = read_json('servers_testnet.json', {})
CHECKPOINTS = read_json('checkpoints_testnet.json', [])
CHECKPOINTS = read_json('checkpoints_testnet.json', {'height': 0})

XPRV_HEADERS = {
'standard': 0x04358394, # tprv
Expand Down Expand Up @@ -135,7 +138,7 @@ class BitcoinRegtest(BitcoinTestnet):
SEGWIT_HRP = "bcrt"
GENESIS = "0f9188f13cb7b2c71f2a335e3a4fc328bf5beb436012afca590b1a11466e2206"
DEFAULT_SERVERS = read_json('servers_regtest.json', {})
CHECKPOINTS = []
CHECKPOINTS = {'height': 0}
LN_DNS_SEEDS = []


Expand All @@ -147,7 +150,7 @@ class BitcoinSimnet(BitcoinTestnet):
SEGWIT_HRP = "sb"
GENESIS = "683e86bd5c6d110d91b94b97137ba6bfe02dbbdb8e3dff722a669b5d69d77af6"
DEFAULT_SERVERS = read_json('servers_regtest.json', {})
CHECKPOINTS = []
CHECKPOINTS = {'height': 0}
LN_DNS_SEEDS = []


Expand Down
151 changes: 131 additions & 20 deletions electrum/interface.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.
import json
import os
import re
import ssl
Expand All @@ -42,13 +43,15 @@
from aiorpcx.rawsocket import RSClient
import certifi

from .util import ignore_exceptions, log_exceptions, bfh, SilentTaskGroup
from .util import ignore_exceptions, log_exceptions, bfh, bh2u, SilentTaskGroup
from . import util
from .crypto import sha256d
from . import x509
from . import pem
from . import version
from .bitcoin import hash_encode
from . import blockchain
from .blockchain import Blockchain
from .blockchain import Blockchain, HeaderChunk, hash_merkle_root
from . import constants
from .i18n import _
from .logging import Logger
Expand Down Expand Up @@ -420,41 +423,147 @@ async def get_certificate(self):
except ValueError:
return None

async def get_block_header(self, height, assert_mode):
# Run manually from console to generate blockchain checkpoints.
# Only use this with a server that you trust!
async def export_purported_checkpoints(self, cp_height, path):
# use lower timeout as we usually have network.bhi_lock here
timeout = self.network.get_network_timeout_seconds(NetworkTimeout.Urgent)

retarget_first_height = cp_height // 2016 * 2016
retarget_last_height = (cp_height+1) // 2016 * 2016 - 1
retarget_last_chunk_index = (cp_height+1) // 2016 - 1

res = await self.session.send_request('blockchain.block.header', [retarget_first_height, cp_height], timeout=timeout)

if 'root' in res and 'branch' in res and 'header' in res:
retarget_first_header = blockchain.deserialize_pure_header(bytes.fromhex(res['header']), retarget_first_height)
retarget_last_chainwork = self.blockchain.get_chainwork(retarget_last_height)
retarget_last_hash = self.blockchain.get_hash(retarget_last_height)
retarget_last_bits = self.blockchain.target_to_bits(self.blockchain.get_target(retarget_last_chunk_index))

# first_timestamp: Timestamp of height // 2016 * 2016
# last_chainwork: Chainwork of (height + 1) // 2016 * 2016 - 1
# last_hash: Hash of (height + 1) // 2016 * 2016 - 1
# last_bits: Bits used in height + 1
cp = {'height': cp_height, 'merkle_root': res['root'],
'first_timestamp': retarget_first_header['timestamp'],
'last_chainwork': retarget_last_chainwork,
'last_hash': retarget_last_hash,
'last_bits': retarget_last_bits}
with open(path, 'w', encoding='utf-8') as f:
f.write(json.dumps(cp, indent=4, sort_keys=True))
else:
raise Exception("Expected checkpoint validation data, did not receive it.")

async def get_block_header(self, height, assert_mode, must_provide_proof=False):
self.logger.info(f'requesting block header {height} in mode {assert_mode}')
# use lower timeout as we usually have network.bhi_lock here
timeout = self.network.get_network_timeout_seconds(NetworkTimeout.Urgent)

cp_height = constants.net.max_checkpoint()
if height > cp_height:
if must_provide_proof:
raise Exception("Can't request a checkpoint proof because requested height is above checkpoint height")
cp_height = 0

res = await self.session.send_request('blockchain.block.header', [height, cp_height], timeout=timeout)
if cp_height != 0:
res = res["header"]
return blockchain.deserialize_full_header(bytes.fromhex(res), height)

proof_was_provided = False
hexheader = None
if 'root' in res and 'branch' in res and 'header' in res:
if cp_height == 0:
raise Exception("Received checkpoint validation data even though it wasn't requested.")

hexheader = res["header"]
self.validate_checkpoint_result(res["root"], res["branch"], hexheader, height)
proof_was_provided = True
elif cp_height != 0:
raise Exception("Expected checkpoint validation data, did not receive it.")
else:
hexheader = res

if proof_was_provided:
return blockchain.deserialize_pure_header(bytes.fromhex(hexheader), height), proof_was_provided
else:
return blockchain.deserialize_full_header(bytes.fromhex(hexheader), height), proof_was_provided

async def request_chunk(self, height, tip=None, *, can_return_early=False):
index = height // 2016
if can_return_early and index in self._requested_chunks:
return

self.logger.info(f"requesting chunk from height {height}")
size = 2016
if tip is not None:
size = min(size, tip - index * 2016 + 1)
size = max(size, 0)
try:
cp_height = constants.net.max_checkpoint()
if index * 2016 + size - 1 > cp_height:
cp_height = 0
self._requested_chunks.add(index)
res = await self.session.send_request('blockchain.block.headers', [index * 2016, size, cp_height])
res, proof_was_provided = await self.request_headers(index * 2016, size)
finally:
try: self._requested_chunks.remove(index)
except KeyError: pass
conn = self.blockchain.connect_chunk(index, res['hex'])
conn = self.blockchain.connect_chunk(index, res['hex'], proof_was_provided)
if not conn:
return conn, 0
return conn, res['count']

async def request_headers(self, height, count):
if count > 2016:
raise Exception("Server does not support requesting more than 2016 consecutive headers")

top_height = height + count - 1
cp_height = constants.net.max_checkpoint()
if top_height > cp_height:
cp_height = 0

res = await self.session.send_request('blockchain.block.headers', [height, count, cp_height])

hexdata = res['hex']
data = bfh(hexdata)
chunk = HeaderChunk(height, data)
actual_header_count = chunk.get_header_count()
# We accept less headers than we asked for, to cover the case where the distance to the tip was unknown.
if actual_header_count > count:
raise Exception("chunk header count too high expected_count={} actual_count={}".format(count, actual_header_count))

proof_was_provided = False
if 'root' in res and 'branch' in res:
if cp_height == 0:
raise Exception("Received checkpoint validation data even though it wasn't requested.")

header_height = height + actual_header_count - 1
header = chunk.get_header_at_height(header_height)
hexheader = bh2u(header)

self.validate_checkpoint_result(res["root"], res["branch"], hexheader, header_height)

blockchain.verify_proven_chunk(height, data)

proof_was_provided = True
elif cp_height != 0:
raise Exception("Expected checkpoint validation data, did not receive it.")

return res, proof_was_provided

def validate_checkpoint_result(self, received_merkle_root, merkle_branch, header, header_height):
'''
header: hex representation of the block header.
merkle_root: hex representation of the server's calculated merkle root.
branch: list of hex representations of the server's calculated merkle root branches.

Raises an exception if the server's proof is incorrect.
'''
expected_merkle_root = constants.net.CHECKPOINTS['merkle_root']

if received_merkle_root != expected_merkle_root:
raise Exception("Sent unexpected merkle root, expected: {}, got: {}".format(expected_merkle_root, received_merkle_root))

header_hash = hash_encode(sha256d(bfh(header)))
proven_merkle_root = hash_merkle_root(merkle_branch, header_hash, header_height, reject_valid_tx=False)
if proven_merkle_root != expected_merkle_root:
raise Exception("Sent incorrect merkle branch, expected: {}, proved: {}".format(constants.net.CHECKPOINTS['merkle_root'], proven_merkle_root))

def is_main_server(self) -> bool:
return self.network.default_server == self.server

Expand Down Expand Up @@ -559,8 +668,9 @@ async def sync_until(self, height, next_height=None):

async def step(self, height, header=None):
assert 0 <= height <= self.tip, (height, self.tip)
proof_was_provided = False
if header is None:
header = await self.get_block_header(height, 'catchup')
header, proof_was_provided = await self.get_block_header(height, 'catchup')

chain = blockchain.check_header(header) if 'mock' not in header else header['mock']['check'](header)
if chain:
Expand All @@ -571,12 +681,12 @@ async def step(self, height, header=None):
# this situation resolves itself on the next block
return 'catchup', height+1

can_connect = blockchain.can_connect(header) if 'mock' not in header else header['mock']['connect'](height)
can_connect = blockchain.can_connect(header, proof_was_provided=proof_was_provided) if 'mock' not in header else header['mock']['connect'](height)
if not can_connect:
self.logger.info(f"can't connect {height}")
height, header, bad, bad_header = await self._search_headers_backwards(height, header)
height, header, bad, bad_header, proof_was_provided = await self._search_headers_backwards(height, header)
chain = blockchain.check_header(header) if 'mock' not in header else header['mock']['check'](header)
can_connect = blockchain.can_connect(header) if 'mock' not in header else header['mock']['connect'](height)
can_connect = blockchain.can_connect(header, proof_was_provided=proof_was_provided) if 'mock' not in header else header['mock']['connect'](height)
assert chain or can_connect
if can_connect:
self.logger.info(f"could connect {height}")
Expand All @@ -599,7 +709,7 @@ async def _search_headers_binary(self, height, bad, bad_header, chain):
assert good < bad, (good, bad)
height = (good + bad) // 2
self.logger.info(f"binary step. good {good}, bad {bad}, height {height}")
header = await self.get_block_header(height, 'binary')
header, proof_was_provided = await self.get_block_header(height, 'binary')
chain = blockchain.check_header(header) if 'mock' not in header else header['mock']['check'](header)
if chain:
self.blockchain = chain if isinstance(chain, Blockchain) else self.blockchain
Expand Down Expand Up @@ -644,14 +754,14 @@ async def _resolve_potential_chain_fork_given_forkpoint(self, good, bad, bad_hea

async def _search_headers_backwards(self, height, header):
async def iterate():
nonlocal height, header
nonlocal height, header, proof_was_provided
checkp = False
if height <= constants.net.max_checkpoint():
height = constants.net.max_checkpoint()
checkp = True
header = await self.get_block_header(height, 'backward')
header, proof_was_provided = await self.get_block_header(height, 'backward')
chain = blockchain.check_header(header) if 'mock' not in header else header['mock']['check'](header)
can_connect = blockchain.can_connect(header) if 'mock' not in header else header['mock']['connect'](height)
can_connect = blockchain.can_connect(header, proof_was_provided=proof_was_provided) if 'mock' not in header else header['mock']['connect'](height)
if chain or can_connect:
return False
if checkp:
Expand All @@ -663,14 +773,15 @@ async def iterate():
with blockchain.blockchains_lock: chains = list(blockchain.blockchains.values())
local_max = max([0] + [x.height() for x in chains]) if 'mock' not in header else float('inf')
height = min(local_max + 1, height - 1)
proof_was_provided = False
while await iterate():
bad, bad_header = height, header
delta = self.tip - height
height = self.tip - 2 * delta

_assert_header_does_not_check_against_any_chain(bad_header)
self.logger.info(f"exiting backward mode at {height}")
return height, header, bad, bad_header
return height, header, bad, bad_header, proof_was_provided

@classmethod
def client_name(cls) -> str:
Expand Down
10 changes: 1 addition & 9 deletions electrum/network.py
Original file line number Diff line number Diff line change
Expand Up @@ -823,7 +823,7 @@ def check_interface_against_healthy_spread_of_connected_servers(self, iface_to_c
async def _init_headers_file(self):
b = blockchain.get_best_chain()
filename = b.path()
length = HEADER_SIZE * len(constants.net.CHECKPOINTS) * 2016
length = HEADER_SIZE * constants.net.max_checkpoint()
if not os.path.exists(filename) or os.path.getsize(filename) < length:
with open(filename, 'wb') as f:
if length > 0:
Expand Down Expand Up @@ -1137,14 +1137,6 @@ async def follow_chain_given_server(self, server_str: str) -> None:
def get_local_height(self):
return self.blockchain().height()

def export_checkpoints(self, path):
"""Run manually to generate blockchain checkpoints.
Kept for console use only.
"""
cp = self.blockchain().get_checkpoints()
with open(path, 'w', encoding='utf-8') as f:
f.write(json.dumps(cp, indent=4))

async def _start(self):
assert not self.main_taskgroup
self.main_taskgroup = main_taskgroup = SilentTaskGroup()
Expand Down
Loading