forked from LBRYCommunity/lbry-sdk
significant performance improvement when creating transactions and with txo spend specifically
This commit is contained in:
parent
39e78ff17e
commit
b208cf6d32
4 changed files with 47 additions and 35 deletions
|
@ -4461,7 +4461,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
@requires(WALLET_COMPONENT)
|
@requires(WALLET_COMPONENT)
|
||||||
async def jsonrpc_txo_spend(
|
async def jsonrpc_txo_spend(
|
||||||
self, account_id=None, wallet_id=None, batch_size=500,
|
self, account_id=None, wallet_id=None, batch_size=100,
|
||||||
include_full_tx=False, preview=False, blocking=False, **kwargs):
|
include_full_tx=False, preview=False, blocking=False, **kwargs):
|
||||||
"""
|
"""
|
||||||
Spend transaction outputs, batching into multiple transactions as necessary.
|
Spend transaction outputs, batching into multiple transactions as necessary.
|
||||||
|
@ -4500,7 +4500,10 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
accounts = [wallet.get_account_or_error(account_id)] if account_id else wallet.accounts
|
accounts = [wallet.get_account_or_error(account_id)] if account_id else wallet.accounts
|
||||||
txos = await self.ledger.get_txos(
|
txos = await self.ledger.get_txos(
|
||||||
wallet=wallet, accounts=accounts, read_only=True,
|
wallet=wallet, accounts=accounts, read_only=True,
|
||||||
**self._constrain_txo_from_kwargs({}, is_not_spent=True, is_my_output=True, **kwargs)
|
no_tx=True, no_channel_info=True,
|
||||||
|
**self._constrain_txo_from_kwargs(
|
||||||
|
{}, is_not_spent=True, is_my_output=True, **kwargs
|
||||||
|
)
|
||||||
)
|
)
|
||||||
txs = []
|
txs = []
|
||||||
while txos:
|
while txos:
|
||||||
|
|
|
@ -965,16 +965,18 @@ class Database(SQLiteMixin):
|
||||||
sql.append("LEFT JOIN txi ON (txi.position=0 AND txi.txid=txo.txid)")
|
sql.append("LEFT JOIN txi ON (txi.position=0 AND txi.txid=txo.txid)")
|
||||||
return await self.db.execute_fetchall(*query(' '.join(sql), **constraints), read_only=read_only)
|
return await self.db.execute_fetchall(*query(' '.join(sql), **constraints), read_only=read_only)
|
||||||
|
|
||||||
async def get_txos(self, wallet=None, no_tx=False, read_only=False, **constraints):
|
async def get_txos(self, wallet=None, no_tx=False, no_channel_info=False, read_only=False, **constraints):
|
||||||
include_is_spent = constraints.get('include_is_spent', False)
|
include_is_spent = constraints.get('include_is_spent', False)
|
||||||
include_is_my_input = constraints.get('include_is_my_input', False)
|
include_is_my_input = constraints.get('include_is_my_input', False)
|
||||||
include_is_my_output = constraints.pop('include_is_my_output', False)
|
include_is_my_output = constraints.pop('include_is_my_output', False)
|
||||||
include_received_tips = constraints.pop('include_received_tips', False)
|
include_received_tips = constraints.pop('include_received_tips', False)
|
||||||
|
|
||||||
select_columns = [
|
select_columns = [
|
||||||
"tx.txid, raw, tx.height, tx.position as tx_position, tx.is_verified, "
|
"tx.txid, tx.height, tx.position as tx_position, tx.is_verified, "
|
||||||
"txo_type, txo.position as txo_position, amount, script"
|
"txo_type, txo.position as txo_position, amount, script"
|
||||||
]
|
]
|
||||||
|
if not no_tx:
|
||||||
|
select_columns.append("raw")
|
||||||
|
|
||||||
my_accounts = {a.public_key.address for a in wallet.accounts} if wallet else set()
|
my_accounts = {a.public_key.address for a in wallet.accounts} if wallet else set()
|
||||||
my_accounts_sql = ""
|
my_accounts_sql = ""
|
||||||
|
@ -1052,6 +1054,7 @@ class Database(SQLiteMixin):
|
||||||
txo.received_tips = row['received_tips']
|
txo.received_tips = row['received_tips']
|
||||||
txos.append(txo)
|
txos.append(txo)
|
||||||
|
|
||||||
|
if not no_channel_info:
|
||||||
channel_ids = set()
|
channel_ids = set()
|
||||||
for txo in txos:
|
for txo in txos:
|
||||||
if txo.is_claim and txo.can_decode_claim:
|
if txo.is_claim and txo.can_decode_claim:
|
||||||
|
|
|
@ -231,7 +231,7 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
async def get_effective_amount_estimators(self, funding_accounts: Iterable[Account]):
|
async def get_effective_amount_estimators(self, funding_accounts: Iterable[Account]):
|
||||||
estimators = []
|
estimators = []
|
||||||
for account in funding_accounts:
|
for account in funding_accounts:
|
||||||
utxos = await account.get_utxos()
|
utxos = await account.get_utxos(no_tx=True, no_channel_info=True)
|
||||||
for utxo in utxos:
|
for utxo in utxos:
|
||||||
estimators.append(utxo.get_estimator(self))
|
estimators.append(utxo.get_estimator(self))
|
||||||
return estimators
|
return estimators
|
||||||
|
|
|
@ -539,6 +539,7 @@ class Transaction:
|
||||||
height: int = -2, position: int = -1, julian_day: int = None) -> None:
|
height: int = -2, position: int = -1, julian_day: int = None) -> None:
|
||||||
self._raw = raw
|
self._raw = raw
|
||||||
self._raw_sans_segwit = None
|
self._raw_sans_segwit = None
|
||||||
|
self._raw_outputs = None
|
||||||
self.is_segwit_flag = 0
|
self.is_segwit_flag = 0
|
||||||
self.witnesses: List[bytes] = []
|
self.witnesses: List[bytes] = []
|
||||||
self.ref = TXRefMutable(self)
|
self.ref = TXRefMutable(self)
|
||||||
|
@ -600,6 +601,7 @@ class Transaction:
|
||||||
def _reset(self):
|
def _reset(self):
|
||||||
self._raw = None
|
self._raw = None
|
||||||
self._raw_sans_segwit = None
|
self._raw_sans_segwit = None
|
||||||
|
self._raw_outputs = None
|
||||||
self.ref.reset()
|
self.ref.reset()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -693,9 +695,7 @@ class Transaction:
|
||||||
stream.write_compact_size(len(self._inputs))
|
stream.write_compact_size(len(self._inputs))
|
||||||
for txin in self._inputs:
|
for txin in self._inputs:
|
||||||
txin.serialize_to(stream)
|
txin.serialize_to(stream)
|
||||||
stream.write_compact_size(len(self._outputs))
|
self._serialize_outputs(stream)
|
||||||
for txout in self._outputs:
|
|
||||||
txout.serialize_to(stream)
|
|
||||||
stream.write_uint32(self.locktime)
|
stream.write_uint32(self.locktime)
|
||||||
return stream.get_bytes()
|
return stream.get_bytes()
|
||||||
|
|
||||||
|
@ -709,13 +709,19 @@ class Transaction:
|
||||||
txin.serialize_to(stream, txin.txo_ref.txo.script.source)
|
txin.serialize_to(stream, txin.txo_ref.txo.script.source)
|
||||||
else:
|
else:
|
||||||
txin.serialize_to(stream, b'')
|
txin.serialize_to(stream, b'')
|
||||||
stream.write_compact_size(len(self._outputs))
|
self._serialize_outputs(stream)
|
||||||
for txout in self._outputs:
|
|
||||||
txout.serialize_to(stream)
|
|
||||||
stream.write_uint32(self.locktime)
|
stream.write_uint32(self.locktime)
|
||||||
stream.write_uint32(self.signature_hash_type(1)) # signature hash type: SIGHASH_ALL
|
stream.write_uint32(self.signature_hash_type(1)) # signature hash type: SIGHASH_ALL
|
||||||
return stream.get_bytes()
|
return stream.get_bytes()
|
||||||
|
|
||||||
|
def _serialize_outputs(self, stream):
|
||||||
|
if self._raw_outputs is None:
|
||||||
|
self._raw_outputs = BCDataStream()
|
||||||
|
self._raw_outputs.write_compact_size(len(self._outputs))
|
||||||
|
for txout in self._outputs:
|
||||||
|
txout.serialize_to(self._raw_outputs)
|
||||||
|
stream.write(self._raw_outputs.get_bytes())
|
||||||
|
|
||||||
def _deserialize(self):
|
def _deserialize(self):
|
||||||
if self._raw is not None:
|
if self._raw is not None:
|
||||||
stream = BCDataStream(self._raw)
|
stream = BCDataStream(self._raw)
|
||||||
|
|
Loading…
Add table
Reference in a new issue