diff --git a/Cargo.lock b/Cargo.lock index 55aa18a9ae..b47619c304 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2562,6 +2562,7 @@ dependencies = [ "futures", "futures-util", "itertools 0.13.0", + "kaspa-addresses", "kaspa-consensus-core", "kaspa-consensus-notify", "kaspa-core", @@ -2676,6 +2677,7 @@ dependencies = [ "faster-hex", "futures", "h2 0.4.6", + "kaspa-addresses", "kaspa-consensus-core", "kaspa-core", "kaspa-notify", diff --git a/cli/src/modules/rpc.rs b/cli/src/modules/rpc.rs index cf6bc6bd20..c721ed6223 100644 --- a/cli/src/modules/rpc.rs +++ b/cli/src/modules/rpc.rs @@ -269,6 +269,21 @@ impl Rpc { let result = rpc.get_current_block_color_call(None, GetCurrentBlockColorRequest { hash }).await?; self.println(&ctx, result); } + RpcApiOps::GetUtxoReturnAddress => { + if argv.is_empty() || argv.len() != 2 { + return Err(Error::custom("Please specify a txid and a accepting_block_daa_score")); + } + + let txid = argv.remove(0); + let txid = RpcHash::from_hex(txid.as_str())?; + + let accepting_block_daa_score = argv.remove(0).parse::()?; + + let result = + rpc.get_utxo_return_address_call(None, GetUtxoReturnAddressRequest { txid, accepting_block_daa_score }).await?; + + self.println(&ctx, result); + } _ => { tprintln!(ctx, "rpc method exists but is not supported by the cli: '{op_str}'\r\n"); return Ok(()); diff --git a/components/addressmanager/src/lib.rs b/components/addressmanager/src/lib.rs index 093323e155..da0f7c1ed1 100644 --- a/components/addressmanager/src/lib.rs +++ b/components/addressmanager/src/lib.rs @@ -596,7 +596,7 @@ mod address_store_with_cache { let target_uniform_dist = Uniform::new(1.0, num_of_buckets as f64).unwrap(); let uniform_cdf = |x: f64| target_uniform_dist.cdf(&x); for _ in 0..num_of_trials { - // The weight sampled expected uniform distibution + // The weight sampled expected uniform distribution let prioritized_address_distribution = am .lock() .iterate_prioritized_random_addresses(HashSet::new()) diff --git a/components/consensusmanager/Cargo.toml b/components/consensusmanager/Cargo.toml index 16f6900871..c9f3645572 100644 --- a/components/consensusmanager/Cargo.toml +++ b/components/consensusmanager/Cargo.toml @@ -14,6 +14,7 @@ duration-string.workspace = true futures-util.workspace = true futures.workspace = true itertools.workspace = true +kaspa-addresses.workspace=true kaspa-consensus-core.workspace = true kaspa-consensus-notify.workspace = true kaspa-core.workspace = true diff --git a/components/consensusmanager/src/lib.rs b/components/consensusmanager/src/lib.rs index 6d31653aab..b1c0b8f61b 100644 --- a/components/consensusmanager/src/lib.rs +++ b/components/consensusmanager/src/lib.rs @@ -213,7 +213,7 @@ impl StagingConsensus { // Drop `prev` so that deletion below succeeds drop(prev); // Staging was committed and is now the active consensus so we can delete - // any pervious, now inactive, consensus entries + // any previous, now inactive, consensus entries self.manager.delete_inactive_consensus_entries(); } diff --git a/components/consensusmanager/src/session.rs b/components/consensusmanager/src/session.rs index 8e0c6e9335..c67caf07d4 100644 --- a/components/consensusmanager/src/session.rs +++ b/components/consensusmanager/src/session.rs @@ -12,7 +12,8 @@ use kaspa_consensus_core::{ header::Header, pruning::{PruningPointProof, PruningPointTrustedData, PruningPointsList}, trusted::{ExternalGhostdagData, TrustedBlock}, - tx::{MutableTransaction, Transaction, TransactionOutpoint, UtxoEntry}, + tx::{MutableTransaction, SignableTransaction, Transaction, TransactionOutpoint, UtxoEntry}, + utxo::utxo_inquirer::UtxoInquirerError, BlockHashSet, BlueWorkType, ChainPath, Hash, }; use kaspa_utils::sync::rwlock::*; @@ -313,6 +314,14 @@ impl ConsensusSessionOwned { self.clone().spawn_blocking(|c| c.get_chain_block_samples()).await } + pub async fn async_get_populated_transaction( + &self, + txid: Hash, + accepting_block_daa_score: u64, + ) -> Result { + self.clone().spawn_blocking(move |c| c.get_populated_transaction(txid, accepting_block_daa_score)).await + } + /// Returns the antipast of block `hash` from the POV of `context`, i.e. `antipast(hash) ∩ past(context)`. /// Since this might be an expensive operation for deep blocks, we allow the caller to specify a limit /// `max_traversal_allowed` on the maximum amount of blocks to traverse for obtaining the answer diff --git a/consensus/Cargo.toml b/consensus/Cargo.toml index 443e591c8a..90aad80e1a 100644 --- a/consensus/Cargo.toml +++ b/consensus/Cargo.toml @@ -18,6 +18,7 @@ faster-hex.workspace = true futures-util.workspace = true indexmap.workspace = true itertools.workspace = true +kaspa-addresses.workspace = true kaspa-consensus-core.workspace = true kaspa-consensus-notify.workspace = true kaspa-consensusmanager.workspace = true diff --git a/consensus/core/src/api/mod.rs b/consensus/core/src/api/mod.rs index 7c244b9148..f8df0c0e14 100644 --- a/consensus/core/src/api/mod.rs +++ b/consensus/core/src/api/mod.rs @@ -19,7 +19,8 @@ use crate::{ header::Header, pruning::{PruningPointProof, PruningPointTrustedData, PruningPointsList, PruningProofMetadata}, trusted::{ExternalGhostdagData, TrustedBlock}, - tx::{MutableTransaction, Transaction, TransactionOutpoint, UtxoEntry}, + tx::{MutableTransaction, SignableTransaction, Transaction, TransactionOutpoint, UtxoEntry}, + utxo::utxo_inquirer::UtxoInquirerError, BlockHashSet, BlueWorkType, ChainPath, }; use kaspa_hashes::Hash; @@ -170,6 +171,12 @@ pub trait ConsensusApi: Send + Sync { unimplemented!() } + /// Returns the fully populated transaction with the given txid which was accepted at the provided accepting_block_daa_score. + /// The argument `accepting_block_daa_score` is expected to be the DAA score of the accepting chain block of `txid`. + fn get_populated_transaction(&self, txid: Hash, accepting_block_daa_score: u64) -> Result { + unimplemented!() + } + fn get_virtual_parents(&self) -> BlockHashSet { unimplemented!() } diff --git a/consensus/core/src/utxo/mod.rs b/consensus/core/src/utxo/mod.rs index 42014856f7..697506cb81 100644 --- a/consensus/core/src/utxo/mod.rs +++ b/consensus/core/src/utxo/mod.rs @@ -1,4 +1,5 @@ pub mod utxo_collection; pub mod utxo_diff; pub mod utxo_error; +pub mod utxo_inquirer; pub mod utxo_view; diff --git a/consensus/core/src/utxo/utxo_inquirer.rs b/consensus/core/src/utxo/utxo_inquirer.rs new file mode 100644 index 0000000000..3aa1000295 --- /dev/null +++ b/consensus/core/src/utxo/utxo_inquirer.rs @@ -0,0 +1,38 @@ +use kaspa_hashes::Hash; +use thiserror::Error; + +#[derive(Error, Debug, Clone)] +pub enum UtxoInquirerError { + #[error("Transaction is already pruned")] + AlreadyPruned, + #[error("Transaction return address is coinbase")] + TxFromCoinbase, + #[error("Transaction not found at given accepting daa score")] + NoTxAtScore, + #[error("Transaction was found but not standard")] + NonStandard, + #[error("Did not find compact header for block hash {0} ")] + MissingCompactHeaderForBlockHash(Hash), + #[error("Did not find containing_acceptance for tx {0} ")] + MissingContainingAcceptanceForTx(Hash), + #[error("Did not find block {0} at block tx store")] + MissingBlockFromBlockTxStore(Hash), + #[error("Did not find index {0} in transactions of block {1}")] + MissingTransactionIndexOfBlock(usize, Hash), + #[error("Expected {0} to match {1} when checking block_transaction_store using array index of transaction")] + UnexpectedTransactionMismatch(Hash, Hash), + #[error("Did not find a utxo diff for chain block {0} ")] + MissingUtxoDiffForChainBlock(Hash), + #[error("Transaction {0} acceptance data must also be in the same block in this case")] + MissingOtherTransactionAcceptanceData(Hash), + #[error("Did not find index for hash {0}")] + MissingIndexForHash(Hash), + #[error("Did not find tip data")] + MissingTipData, + #[error("Did not find a hash at index {0} ")] + MissingHashAtIndex(u64), + #[error("Did not find acceptance data for chain block {0}")] + MissingAcceptanceDataForChainBlock(Hash), + #[error("Utxo entry is not filled")] + UnfilledUtxoEntry, +} diff --git a/consensus/src/consensus/mod.rs b/consensus/src/consensus/mod.rs index 99719d4ac2..21e5bf5573 100644 --- a/consensus/src/consensus/mod.rs +++ b/consensus/src/consensus/mod.rs @@ -66,7 +66,8 @@ use kaspa_consensus_core::{ network::NetworkType, pruning::{PruningPointProof, PruningPointTrustedData, PruningPointsList, PruningProofMetadata}, trusted::{ExternalGhostdagData, TrustedBlock}, - tx::{MutableTransaction, Transaction, TransactionOutpoint, UtxoEntry}, + tx::{MutableTransaction, SignableTransaction, Transaction, TransactionOutpoint, UtxoEntry}, + utxo::utxo_inquirer::UtxoInquirerError, BlockHashSet, BlueWorkType, ChainPath, HashMapCustomHasher, }; use kaspa_consensus_notify::root::ConsensusNotificationRoot; @@ -687,6 +688,12 @@ impl ConsensusApi for Consensus { sample_headers } + fn get_populated_transaction(&self, txid: Hash, accepting_block_daa_score: u64) -> Result { + // We need consistency between the pruning_point_store, utxo_diffs_store, block_transactions_store, selected chain and headers store reads + let _guard = self.pruning_lock.blocking_read(); + self.virtual_processor.get_populated_transaction(txid, accepting_block_daa_score, self.get_source()) + } + fn get_virtual_parents(&self) -> BlockHashSet { self.lkg_virtual_state.load().parents.iter().copied().collect() } diff --git a/consensus/src/pipeline/virtual_processor/mod.rs b/consensus/src/pipeline/virtual_processor/mod.rs index a35dec6856..fd3756f3ae 100644 --- a/consensus/src/pipeline/virtual_processor/mod.rs +++ b/consensus/src/pipeline/virtual_processor/mod.rs @@ -1,5 +1,6 @@ pub mod errors; mod processor; +mod utxo_inquirer; mod utxo_validation; pub use processor::*; pub mod test_block_builder; diff --git a/consensus/src/pipeline/virtual_processor/processor.rs b/consensus/src/pipeline/virtual_processor/processor.rs index 914e0a327d..21c347c4df 100644 --- a/consensus/src/pipeline/virtual_processor/processor.rs +++ b/consensus/src/pipeline/virtual_processor/processor.rs @@ -156,7 +156,7 @@ pub struct VirtualStateProcessor { pub(super) block_window_cache_for_past_median_time: Arc, // Pruning lock - pruning_lock: SessionLock, + pub(super) pruning_lock: SessionLock, // Notifier notification_root: Arc, diff --git a/consensus/src/pipeline/virtual_processor/utxo_inquirer.rs b/consensus/src/pipeline/virtual_processor/utxo_inquirer.rs new file mode 100644 index 0000000000..617c1026bd --- /dev/null +++ b/consensus/src/pipeline/virtual_processor/utxo_inquirer.rs @@ -0,0 +1,189 @@ +use std::{cmp, sync::Arc}; + +use kaspa_consensus_core::{ + acceptance_data::AcceptanceData, + tx::{SignableTransaction, Transaction, UtxoEntry}, + utxo::{utxo_diff::ImmutableUtxoDiff, utxo_inquirer::UtxoInquirerError}, +}; +use kaspa_core::{trace, warn}; +use kaspa_hashes::Hash; + +use crate::model::stores::{ + acceptance_data::AcceptanceDataStoreReader, block_transactions::BlockTransactionsStoreReader, headers::HeaderStoreReader, + selected_chain::SelectedChainStoreReader, utxo_diffs::UtxoDiffsStoreReader, +}; + +use super::VirtualStateProcessor; + +impl VirtualStateProcessor { + /// Returns the fully populated transaction with the given txid which was accepted at the provided accepting_block_daa_score. + /// The argument `accepting_block_daa_score` is expected to be the DAA score of the accepting chain block of `txid`. + /// + /// *Assumed to be called under the pruning read lock.* + pub fn get_populated_transaction( + &self, + txid: Hash, + accepting_block_daa_score: u64, + source_hash: Hash, + ) -> Result { + let source_daa_score = self + .headers_store + .get_compact_header_data(source_hash) + .map(|compact_header| compact_header.daa_score) + .map_err(|_| UtxoInquirerError::MissingCompactHeaderForBlockHash(source_hash))?; + + if accepting_block_daa_score < source_daa_score { + // Early exit if target daa score is lower than that of pruning point's daa score: + return Err(UtxoInquirerError::AlreadyPruned); + } + + let (matching_chain_block_hash, acceptance_data) = + self.find_accepting_chain_block_hash_at_daa_score(accepting_block_daa_score, source_hash)?; + + // Expected to never fail, since we found the acceptance data and therefore there must be matching diff + let utxo_diff = self + .utxo_diffs_store + .get(matching_chain_block_hash) + .map_err(|_| UtxoInquirerError::MissingUtxoDiffForChainBlock(matching_chain_block_hash))?; + + let tx = self.find_tx_from_acceptance_data(txid, &acceptance_data)?; + + let mut populated_tx = SignableTransaction::new(tx); + + let removed_diffs = utxo_diff.removed(); + + populated_tx.tx.inputs.iter().enumerate().for_each(|(index, input)| { + let filled_utxo = if let Some(utxo_entry) = removed_diffs.get(&input.previous_outpoint) { + Some(utxo_entry.clone().to_owned()) + } else { + // This handles this rare scenario: + // - UTXO0 is spent by TX1 and creates UTXO1 + // - UTXO1 is spent by TX2 and creates UTXO2 + // - A chain block happens to accept both of these + // In this case, removed_diff wouldn't contain the outpoint of the created-and-immediately-spent UTXO + // so we use the transaction (which also has acceptance data in this block) and look at its outputs + let other_txid = input.previous_outpoint.transaction_id; + let other_tx = self.find_tx_from_acceptance_data(other_txid, &acceptance_data).unwrap(); + let output = &other_tx.outputs[input.previous_outpoint.index as usize]; + let utxo_entry = + UtxoEntry::new(output.value, output.script_public_key.clone(), accepting_block_daa_score, other_tx.is_coinbase()); + Some(utxo_entry) + }; + + populated_tx.entries[index] = filled_utxo; + }); + + Ok(populated_tx) + } + + /// Find the accepting chain block hash at the given DAA score by binary searching + /// through selected chain store using indexes. + /// This method assumes that local caller have acquired the pruning read lock to guarantee + /// consistency between reads on the selected_chain_store and headers_store (as well as + /// other stores outside). If no such lock is acquired, this method tries to find + /// the accepting chain block hash on a best effort basis (may fail if parts of the data + /// are pruned between two sequential calls) + fn find_accepting_chain_block_hash_at_daa_score( + &self, + target_daa_score: u64, + source_hash: Hash, + ) -> Result<(Hash, Arc), UtxoInquirerError> { + let sc_read = self.selected_chain_store.read(); + + let source_index = sc_read.get_by_hash(source_hash).map_err(|_| UtxoInquirerError::MissingIndexForHash(source_hash))?; + let (tip_index, tip_hash) = sc_read.get_tip().map_err(|_| UtxoInquirerError::MissingTipData)?; + let tip_daa_score = self + .headers_store + .get_compact_header_data(tip_hash) + .map(|tip| tip.daa_score) + .map_err(|_| UtxoInquirerError::MissingCompactHeaderForBlockHash(tip_hash))?; + + // For a chain segment it holds that len(segment) <= daa_score(segment end) - daa_score(segment start). This is true + // because each chain block increases the daa score by at least one. Hence we can lower bound our search by high index + // minus the daa score gap as done below + let mut low_index = tip_index.saturating_sub(tip_daa_score.saturating_sub(target_daa_score)).max(source_index); + let mut high_index = tip_index; + + let matching_chain_block_hash = loop { + // Binary search for the chain block that matches the target_daa_score + // 0. Get the mid point index + let mid = low_index + (high_index - low_index) / 2; + + // 1. Get the chain block hash at that index. Error if we cannot find a hash at that index + let hash = sc_read.get_by_index(mid).map_err(|_| { + trace!("Did not find a hash at index {}", mid); + UtxoInquirerError::MissingHashAtIndex(mid) + })?; + + // 2. Get the compact header so we have access to the daa_score. Error if we cannot find the header + let compact_header = self.headers_store.get_compact_header_data(hash).map_err(|_| { + trace!("Did not find a compact header with hash {}", hash); + UtxoInquirerError::MissingCompactHeaderForBlockHash(hash) + })?; + + // 3. Compare block daa score to our target + match compact_header.daa_score.cmp(&target_daa_score) { + cmp::Ordering::Equal => { + // We found the chain block we need + break hash; + } + cmp::Ordering::Greater => { + high_index = mid - 1; + } + cmp::Ordering::Less => { + low_index = mid + 1; + } + } + + if low_index > high_index { + return Err(UtxoInquirerError::NoTxAtScore); + } + }; + + let acceptance_data = self + .acceptance_data_store + .get(matching_chain_block_hash) + .map_err(|_| UtxoInquirerError::MissingAcceptanceDataForChainBlock(matching_chain_block_hash))?; + + Ok((matching_chain_block_hash, acceptance_data)) + } + + /// Finds a transaction's containing block hash and index within block through + /// the accepting block acceptance data + fn find_containing_block_and_index_from_acceptance_data( + &self, + txid: Hash, + acceptance_data: &AcceptanceData, + ) -> Option<(Hash, usize)> { + acceptance_data.iter().find_map(|mbad| { + let tx_arr_index = + mbad.accepted_transactions.iter().find_map(|tx| (tx.transaction_id == txid).then_some(tx.index_within_block as usize)); + tx_arr_index.map(|index| (mbad.block_hash, index)) + }) + } + + /// Finds a transaction through the accepting block acceptance data (and using indexed info therein for + /// finding the tx in the block transactions store) + fn find_tx_from_acceptance_data(&self, txid: Hash, acceptance_data: &AcceptanceData) -> Result { + let (containing_block, index) = self + .find_containing_block_and_index_from_acceptance_data(txid, acceptance_data) + .ok_or(UtxoInquirerError::MissingContainingAcceptanceForTx(txid))?; + + let tx = self + .block_transactions_store + .get(containing_block) + .map_err(|_| UtxoInquirerError::MissingBlockFromBlockTxStore(containing_block)) + .and_then(|block_txs| { + block_txs.get(index).cloned().ok_or(UtxoInquirerError::MissingTransactionIndexOfBlock(index, containing_block)) + })?; + + if tx.id() != txid { + // Should never happen, but do a sanity check. This would mean something went wrong with storing block transactions. + // Sanity check is necessary to guarantee that this function will never give back a wrong address (err on the side of not found) + warn!("Expected {} to match {} when checking block_transaction_store using array index of transaction", tx.id(), txid); + return Err(UtxoInquirerError::UnexpectedTransactionMismatch(tx.id(), txid)); + } + + Ok(tx) + } +} diff --git a/rpc/core/src/api/ops.rs b/rpc/core/src/api/ops.rs index 26ca356eb0..4541ddc56d 100644 --- a/rpc/core/src/api/ops.rs +++ b/rpc/core/src/api/ops.rs @@ -136,6 +136,8 @@ pub enum RpcApiOps { GetFeeEstimateExperimental = 148, /// Block color determination by iterating DAG. GetCurrentBlockColor = 149, + /// Get UTXO Return Addresses + GetUtxoReturnAddress = 150, } impl RpcApiOps { diff --git a/rpc/core/src/api/rpc.rs b/rpc/core/src/api/rpc.rs index cadc9e00cd..d9011c167e 100644 --- a/rpc/core/src/api/rpc.rs +++ b/rpc/core/src/api/rpc.rs @@ -438,6 +438,18 @@ pub trait RpcApi: Sync + Send + AnySync { request: GetDaaScoreTimestampEstimateRequest, ) -> RpcResult; + async fn get_utxo_return_address(&self, txid: RpcHash, accepting_block_daa_score: u64) -> RpcResult { + Ok(self + .get_utxo_return_address_call(None, GetUtxoReturnAddressRequest { txid, accepting_block_daa_score }) + .await? + .return_address) + } + async fn get_utxo_return_address_call( + &self, + _connection: Option<&DynRpcConnection>, + request: GetUtxoReturnAddressRequest, + ) -> RpcResult; + // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // Fee estimation API diff --git a/rpc/core/src/error.rs b/rpc/core/src/error.rs index 0e2bfee225..54763c71ba 100644 --- a/rpc/core/src/error.rs +++ b/rpc/core/src/error.rs @@ -2,7 +2,7 @@ //! [`RpcError`] enum used by RPC primitives. //! -use kaspa_consensus_core::{subnets::SubnetworkConversionError, tx::TransactionId}; +use kaspa_consensus_core::{subnets::SubnetworkConversionError, tx::TransactionId, utxo::utxo_inquirer::UtxoInquirerError}; use kaspa_utils::networking::IpAddress; use std::{net::AddrParseError, num::TryFromIntError}; use thiserror::Error; @@ -134,6 +134,9 @@ pub enum RpcError { #[error(transparent)] ConsensusClient(#[from] kaspa_consensus_client::error::Error), + + #[error("utxo return address could not be found -> {0}")] + UtxoReturnAddressNotFound(UtxoInquirerError), } impl From for RpcError { diff --git a/rpc/core/src/model/message.rs b/rpc/core/src/model/message.rs index ba8d6abf76..cb663c394a 100644 --- a/rpc/core/src/model/message.rs +++ b/rpc/core/src/model/message.rs @@ -2666,6 +2666,69 @@ impl Deserializer for GetCurrentBlockColorResponse { } } +#[derive(Clone, Debug, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct GetUtxoReturnAddressRequest { + pub txid: RpcHash, + pub accepting_block_daa_score: u64, +} + +impl GetUtxoReturnAddressRequest { + pub fn new(txid: RpcHash, accepting_block_daa_score: u64) -> Self { + Self { txid, accepting_block_daa_score } + } +} + +impl Serializer for GetUtxoReturnAddressRequest { + fn serialize(&self, writer: &mut W) -> std::io::Result<()> { + store!(u16, &1, writer)?; + store!(RpcHash, &self.txid, writer)?; + store!(u64, &self.accepting_block_daa_score, writer)?; + + Ok(()) + } +} + +impl Deserializer for GetUtxoReturnAddressRequest { + fn deserialize(reader: &mut R) -> std::io::Result { + let _version = load!(u16, reader)?; + let txid = load!(RpcHash, reader)?; + let accepting_block_daa_score = load!(u64, reader)?; + + Ok(Self { txid, accepting_block_daa_score }) + } +} + +#[derive(Clone, Debug, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct GetUtxoReturnAddressResponse { + pub return_address: RpcAddress, +} + +impl GetUtxoReturnAddressResponse { + pub fn new(return_address: RpcAddress) -> Self { + Self { return_address } + } +} + +impl Serializer for GetUtxoReturnAddressResponse { + fn serialize(&self, writer: &mut W) -> std::io::Result<()> { + store!(u16, &1, writer)?; + store!(RpcAddress, &self.return_address, writer)?; + + Ok(()) + } +} + +impl Deserializer for GetUtxoReturnAddressResponse { + fn deserialize(reader: &mut R) -> std::io::Result { + let _version = load!(u16, reader)?; + let return_address = load!(RpcAddress, reader)?; + + Ok(Self { return_address }) + } +} + // ---------------------------------------------------------------------------- // Subscriptions & notifications // ---------------------------------------------------------------------------- diff --git a/rpc/grpc/client/src/lib.rs b/rpc/grpc/client/src/lib.rs index b7e53bb5e1..04e37b5fe6 100644 --- a/rpc/grpc/client/src/lib.rs +++ b/rpc/grpc/client/src/lib.rs @@ -276,6 +276,7 @@ impl RpcApi for GrpcClient { route!(get_fee_estimate_call, GetFeeEstimate); route!(get_fee_estimate_experimental_call, GetFeeEstimateExperimental); route!(get_current_block_color_call, GetCurrentBlockColor); + route!(get_utxo_return_address_call, GetUtxoReturnAddress); // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // Notification API diff --git a/rpc/grpc/core/Cargo.toml b/rpc/grpc/core/Cargo.toml index 2edc10b600..61734df9e3 100644 --- a/rpc/grpc/core/Cargo.toml +++ b/rpc/grpc/core/Cargo.toml @@ -10,6 +10,7 @@ license.workspace = true repository.workspace = true [dependencies] +kaspa-addresses.workspace = true kaspa-consensus-core.workspace = true kaspa-core.workspace = true kaspa-notify.workspace = true diff --git a/rpc/grpc/core/proto/messages.proto b/rpc/grpc/core/proto/messages.proto index 2d6310d9e3..ccb2798b67 100644 --- a/rpc/grpc/core/proto/messages.proto +++ b/rpc/grpc/core/proto/messages.proto @@ -65,6 +65,7 @@ message KaspadRequest { GetFeeEstimateRequestMessage getFeeEstimateRequest = 1106; GetFeeEstimateExperimentalRequestMessage getFeeEstimateExperimentalRequest = 1108; GetCurrentBlockColorRequestMessage getCurrentBlockColorRequest = 1110; + GetUtxoReturnAddressRequestMessage GetUtxoReturnAddressRequest = 1112; } } @@ -130,6 +131,7 @@ message KaspadResponse { GetFeeEstimateResponseMessage getFeeEstimateResponse = 1107; GetFeeEstimateExperimentalResponseMessage getFeeEstimateExperimentalResponse = 1109; GetCurrentBlockColorResponseMessage getCurrentBlockColorResponse = 1111; + GetUtxoReturnAddressResponseMessage GetUtxoReturnAddressResponse = 1113; } } diff --git a/rpc/grpc/core/proto/rpc.proto b/rpc/grpc/core/proto/rpc.proto index e218681b65..4c36150ad3 100644 --- a/rpc/grpc/core/proto/rpc.proto +++ b/rpc/grpc/core/proto/rpc.proto @@ -904,7 +904,7 @@ message GetDaaScoreTimestampEstimateRequestMessage { repeated uint64 daaScores = 1; } -message GetDaaScoreTimestampEstimateResponseMessage{ +message GetDaaScoreTimestampEstimateResponseMessage { repeated uint64 timestamps = 1; RPCError error = 1000; } @@ -974,3 +974,13 @@ message GetCurrentBlockColorResponseMessage { RPCError error = 1000; } + +message GetUtxoReturnAddressRequestMessage { + string txid = 1; + uint64 accepting_block_daa_score = 2; +} + +message GetUtxoReturnAddressResponseMessage { + string return_address = 1; + RPCError error = 1000; +} diff --git a/rpc/grpc/core/src/convert/kaspad.rs b/rpc/grpc/core/src/convert/kaspad.rs index c3411545cc..7243fd401a 100644 --- a/rpc/grpc/core/src/convert/kaspad.rs +++ b/rpc/grpc/core/src/convert/kaspad.rs @@ -63,6 +63,7 @@ pub mod kaspad_request_convert { impl_into_kaspad_request!(GetFeeEstimate); impl_into_kaspad_request!(GetFeeEstimateExperimental); impl_into_kaspad_request!(GetCurrentBlockColor); + impl_into_kaspad_request!(GetUtxoReturnAddress); impl_into_kaspad_request!(NotifyBlockAdded); impl_into_kaspad_request!(NotifyNewBlockTemplate); @@ -200,6 +201,7 @@ pub mod kaspad_response_convert { impl_into_kaspad_response!(GetFeeEstimate); impl_into_kaspad_response!(GetFeeEstimateExperimental); impl_into_kaspad_response!(GetCurrentBlockColor); + impl_into_kaspad_response!(GetUtxoReturnAddress); impl_into_kaspad_notify_response!(NotifyBlockAdded); impl_into_kaspad_notify_response!(NotifyNewBlockTemplate); diff --git a/rpc/grpc/core/src/convert/message.rs b/rpc/grpc/core/src/convert/message.rs index 67ac60650c..c92e824ed0 100644 --- a/rpc/grpc/core/src/convert/message.rs +++ b/rpc/grpc/core/src/convert/message.rs @@ -19,7 +19,8 @@ //! The SubmitBlockResponse is a notable exception to this general rule. use crate::protowire::{self, submit_block_response_message::RejectReason}; -use kaspa_consensus_core::network::NetworkId; +use kaspa_addresses::Address; +use kaspa_consensus_core::{network::NetworkId, Hash}; use kaspa_core::debug; use kaspa_notify::subscription::Command; use kaspa_rpc_core::{ @@ -430,6 +431,16 @@ from!(item: RpcResult<&kaspa_rpc_core::GetCurrentBlockColorResponse>, protowire: Self { blue: item.blue, error: None } }); +from!(item: &kaspa_rpc_core::GetUtxoReturnAddressRequest, protowire::GetUtxoReturnAddressRequestMessage, { + Self { + txid: item.txid.to_string(), + accepting_block_daa_score: item.accepting_block_daa_score + } +}); +from!(item: RpcResult<&kaspa_rpc_core::GetUtxoReturnAddressResponse>, protowire::GetUtxoReturnAddressResponseMessage, { + Self { return_address: item.return_address.address_to_string(), error: None } +}); + from!(&kaspa_rpc_core::PingRequest, protowire::PingRequestMessage); from!(RpcResult<&kaspa_rpc_core::PingResponse>, protowire::PingResponseMessage); @@ -916,6 +927,15 @@ try_from!(item: &protowire::GetCurrentBlockColorResponseMessage, RpcResult, { + Self { return_address: Address::try_from(item.return_address.clone())? } +}); try_from!(&protowire::PingRequestMessage, kaspa_rpc_core::PingRequest); try_from!(&protowire::PingResponseMessage, RpcResult); diff --git a/rpc/grpc/core/src/ops.rs b/rpc/grpc/core/src/ops.rs index f3bc12c829..223774c74c 100644 --- a/rpc/grpc/core/src/ops.rs +++ b/rpc/grpc/core/src/ops.rs @@ -87,6 +87,7 @@ pub enum KaspadPayloadOps { GetFeeEstimate, GetFeeEstimateExperimental, GetCurrentBlockColor, + GetUtxoReturnAddress, // Subscription commands for starting/stopping notifications NotifyBlockAdded, diff --git a/rpc/grpc/server/src/request_handler/factory.rs b/rpc/grpc/server/src/request_handler/factory.rs index b6a5b4476f..9fec86e476 100644 --- a/rpc/grpc/server/src/request_handler/factory.rs +++ b/rpc/grpc/server/src/request_handler/factory.rs @@ -81,6 +81,7 @@ impl Factory { GetFeeEstimate, GetFeeEstimateExperimental, GetCurrentBlockColor, + GetUtxoReturnAddress, NotifyBlockAdded, NotifyNewBlockTemplate, NotifyFinalityConflict, diff --git a/rpc/grpc/server/src/tests/rpc_core_mock.rs b/rpc/grpc/server/src/tests/rpc_core_mock.rs index dd6de46d2a..ada801850e 100644 --- a/rpc/grpc/server/src/tests/rpc_core_mock.rs +++ b/rpc/grpc/server/src/tests/rpc_core_mock.rs @@ -362,6 +362,14 @@ impl RpcApi for RpcCoreMock { Err(RpcError::NotImplemented) } + async fn get_utxo_return_address_call( + &self, + _connection: Option<&DynRpcConnection>, + _request: GetUtxoReturnAddressRequest, + ) -> RpcResult { + Err(RpcError::NotImplemented) + } + // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // Notification API diff --git a/rpc/service/src/service.rs b/rpc/service/src/service.rs index d75ff770b0..c8c40c7707 100644 --- a/rpc/service/src/service.rs +++ b/rpc/service/src/service.rs @@ -7,6 +7,7 @@ use crate::service::NetworkType::{Mainnet, Testnet}; use async_trait::async_trait; use kaspa_consensus_core::api::counters::ProcessingCounters; use kaspa_consensus_core::errors::block::RuleError; +use kaspa_consensus_core::utxo::utxo_inquirer::UtxoInquirerError; use kaspa_consensus_core::{ block::Block, coinbase::MinerData, @@ -794,6 +795,33 @@ NOTE: This error usually indicates an RPC conversion error between the node and } } + async fn get_utxo_return_address_call( + &self, + _connection: Option<&DynRpcConnection>, + request: GetUtxoReturnAddressRequest, + ) -> RpcResult { + let session = self.consensus_manager.consensus().session().await; + + match session.async_get_populated_transaction(request.txid, request.accepting_block_daa_score).await { + Ok(tx) => { + if tx.tx.inputs.is_empty() || tx.entries.is_empty() { + return Err(RpcError::UtxoReturnAddressNotFound(UtxoInquirerError::TxFromCoinbase)); + } + + if let Some(utxo_entry) = &tx.entries[0] { + if let Ok(address) = extract_script_pub_key_address(&utxo_entry.script_public_key, self.config.prefix()) { + Ok(GetUtxoReturnAddressResponse { return_address: address }) + } else { + Err(RpcError::UtxoReturnAddressNotFound(UtxoInquirerError::NonStandard)) + } + } else { + Err(RpcError::UtxoReturnAddressNotFound(UtxoInquirerError::UnfilledUtxoEntry)) + } + } + Err(error) => return Err(RpcError::UtxoReturnAddressNotFound(error)), + } + } + async fn ping_call(&self, _connection: Option<&DynRpcConnection>, _: PingRequest) -> RpcResult { Ok(PingResponse {}) } diff --git a/rpc/wrpc/client/src/client.rs b/rpc/wrpc/client/src/client.rs index 3ac04fa984..f22bcf6255 100644 --- a/rpc/wrpc/client/src/client.rs +++ b/rpc/wrpc/client/src/client.rs @@ -644,6 +644,7 @@ impl RpcApi for KaspaRpcClient { GetSubnetwork, GetSyncStatus, GetSystemInfo, + GetUtxoReturnAddress, GetUtxosByAddresses, GetVirtualChainFromBlock, ResolveFinalityConflict, diff --git a/rpc/wrpc/server/src/router.rs b/rpc/wrpc/server/src/router.rs index 4d0e206259..b4c74a3374 100644 --- a/rpc/wrpc/server/src/router.rs +++ b/rpc/wrpc/server/src/router.rs @@ -47,13 +47,14 @@ impl Router { GetCurrentBlockColor, GetCoinSupply, GetConnectedPeerInfo, + GetDaaScoreTimestampEstimate, + GetUtxoReturnAddress, GetCurrentNetwork, GetDaaScoreTimestampEstimate, GetFeeEstimate, GetFeeEstimateExperimental, GetHeaders, GetInfo, - GetInfo, GetMempoolEntries, GetMempoolEntriesByAddresses, GetMempoolEntry, diff --git a/testing/integration/src/daemon_integration_tests.rs b/testing/integration/src/daemon_integration_tests.rs index 460cf049c3..25471cab13 100644 --- a/testing/integration/src/daemon_integration_tests.rs +++ b/testing/integration/src/daemon_integration_tests.rs @@ -274,7 +274,8 @@ async fn daemon_utxos_propagation_test() { clients.iter().for_each(|x| x.utxos_changed_listener().unwrap().drain()); clients.iter().for_each(|x| x.virtual_daa_score_changed_listener().unwrap().drain()); - // Spend some coins + // Spend some coins - sending funds from miner address to user address + // The transaction here is later used to verify utxo return address RPC const NUMBER_INPUTS: u64 = 2; const NUMBER_OUTPUTS: u64 = 2; const TX_AMOUNT: u64 = SIMNET_PARAMS.pre_deflationary_phase_base_subsidy * (NUMBER_INPUTS * 5 - 1) / 5; @@ -324,6 +325,23 @@ async fn daemon_utxos_propagation_test() { assert_eq!(user_balance, TX_AMOUNT); } + // UTXO Return Address Test + // Mine another block to accept the transactions from the previous block + // The tx above is sending from miner address to user address + mine_block(blank_address.clone(), &rpc_client1, &clients).await; + let new_utxos = rpc_client1.get_utxos_by_addresses(vec![user_address]).await.unwrap(); + let new_utxo = new_utxos + .iter() + .find(|utxo| utxo.outpoint.transaction_id == transaction.id()) + .expect("Did not find a utxo for the tx we just created but expected to"); + + let utxo_return_address = rpc_client1 + .get_utxo_return_address(new_utxo.outpoint.transaction_id, new_utxo.utxo_entry.block_daa_score) + .await + .expect("We just created the tx and utxo here"); + + assert_eq!(miner_address, utxo_return_address); + // Terminate multi-listener clients for x in clients.iter() { x.disconnect().await.unwrap(); diff --git a/testing/integration/src/rpc_tests.rs b/testing/integration/src/rpc_tests.rs index 3c4df601b3..3ca619423a 100644 --- a/testing/integration/src/rpc_tests.rs +++ b/testing/integration/src/rpc_tests.rs @@ -656,6 +656,23 @@ async fn sanity_test() { }) } + KaspadPayloadOps::GetUtxoReturnAddress => { + let rpc_client = client.clone(); + tst!(op, { + let results = rpc_client.get_utxo_return_address(RpcHash::from_bytes([0; 32]), 1000).await; + + assert!(results.is_err_and(|err| { + match err { + kaspa_rpc_core::RpcError::General(msg) => { + info!("Expected error message: {}", msg); + true + } + _ => false, + } + })); + }) + } + KaspadPayloadOps::NotifyBlockAdded => { let rpc_client = client.clone(); let id = listener_id; diff --git a/wallet/core/src/tests/rpc_core_mock.rs b/wallet/core/src/tests/rpc_core_mock.rs index 4d10cdd9b1..529fffffe8 100644 --- a/wallet/core/src/tests/rpc_core_mock.rs +++ b/wallet/core/src/tests/rpc_core_mock.rs @@ -379,6 +379,14 @@ impl RpcApi for RpcCoreMock { Err(RpcError::NotImplemented) } + async fn get_utxo_return_address_call( + &self, + _connection: Option<&DynRpcConnection>, + _request: GetUtxoReturnAddressRequest, + ) -> RpcResult { + Err(RpcError::NotImplemented) + } + // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // Notification API diff --git a/wasm/examples/nodejs/javascript/transactions/generator.js b/wasm/examples/nodejs/javascript/transactions/generator.js index ca6819f086..fa3f4f3aeb 100644 --- a/wasm/examples/nodejs/javascript/transactions/generator.js +++ b/wasm/examples/nodejs/javascript/transactions/generator.js @@ -67,7 +67,7 @@ const { encoding, networkId, address : destinationAddress } = require("../utils" // to the change address. // // If the requested amount is greater than the Kaspa - // transactoin mass, the Generator will create multiple + // transaction mass, the Generator will create multiple // transactions where each transaction will forward // UTXOs to the change address, until the requested // amount is reached. It will then create a final