From 3bcc146a44cd9e191c57a181a7fdcdb03006f81a Mon Sep 17 00:00:00 2001 From: Tobin Harding Date: Wed, 9 Mar 2022 12:27:12 +1100 Subject: [PATCH 1/4] Improve docs: encode_signing_data_to/signature_hash The two methods `encode_signing_data_to` and `signature_hash` use the same docs (one is a public helper for the other). The docs have gotten a bit stale (refer to deprecated types). Instead of duplicating all the text, add a statement pointing readers from the docs of `signature_hash` to the docs on `encode_signing_data_to`. --- src/blockdata/transaction.rs | 35 ++++++++++++++--------------------- 1 file changed, 14 insertions(+), 21 deletions(-) diff --git a/src/blockdata/transaction.rs b/src/blockdata/transaction.rs index 244356387..715701c3b 100644 --- a/src/blockdata/transaction.rs +++ b/src/blockdata/transaction.rs @@ -310,20 +310,23 @@ impl Transaction { } /// Encodes the signing data from which a signature hash for a given input index with a given - /// sighash flag can be computed. To actually produce a scriptSig, this hash needs to be run - /// through an ECDSA signer, the SigHashType appended to the resulting sig, and a script - /// written around this, but this is the general (and hard) part. + /// sighash flag can be computed. /// - /// The `sighash_type` supports arbitrary `u32` value, instead of just [`SigHashType`], - /// because internally 4 bytes are being hashed, even though only lowest byte - /// is appended to signature in a transaction. + /// To actually produce a scriptSig, this hash needs to be run through an ECDSA signer, the + /// [`EcdsaSigHashType`] appended to the resulting sig, and a script written around this, but + /// this is the general (and hard) part. /// - /// *Warning* This does NOT attempt to support OP_CODESEPARATOR. In general this would require - /// evaluating `script_pubkey` to determine which separators get evaluated and which don't, - /// which we don't have the information to determine. + /// The `sighash_type` supports an arbitrary `u32` value, instead of just [`EcdsaSigHashType`], + /// because internally 4 bytes are being hashed, even though only lowest byte is appended to + /// signature in a transaction. + /// + /// **Warning**: This does NOT attempt to support OP_CODESEPARATOR. In general this would + /// require evaluating `script_pubkey` to determine which separators get evaluated and which + /// don't, which we don't have the information to determine. /// /// # Panics - /// Panics if `input_index` is greater than or equal to `self.input.len()` + /// + /// If `input_index` is out of bounds (greater than or equal to `self.input.len()`). pub fn encode_signing_data_to>( &self, mut writer: Write, @@ -392,18 +395,8 @@ impl Transaction { } /// Computes a signature hash for a given input index with a given sighash flag. - /// To actually produce a scriptSig, this hash needs to be run through an - /// ECDSA signer, the SigHashType appended to the resulting sig, and a - /// script written around this, but this is the general (and hard) part. - /// - /// *Warning* This does NOT attempt to support OP_CODESEPARATOR. In general - /// this would require evaluating `script_pubkey` to determine which separators - /// get evaluated and which don't, which we don't have the information to - /// determine. - /// - /// # Panics - /// Panics if `input_index` is greater than or equal to `self.input.len()` /// + /// Please refer [`Self::encode_signing_data_to`] for full documentation of this method. pub fn signature_hash( &self, input_index: usize, From 6a0ec1ac4743873983cc8dc9a5fb5a300306e6fa Mon Sep 17 00:00:00 2001 From: Tobin Harding Date: Wed, 9 Mar 2022 12:42:42 +1100 Subject: [PATCH 2/4] Remove redundant _eq `assert!` already checks a boolean, it is redundant to use `assert_eq!` and pass in `true`. Remove redundant usage of `assert_eq!(foo, true)`. --- src/blockdata/transaction.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/blockdata/transaction.rs b/src/blockdata/transaction.rs index 715701c3b..fcb3f0224 100644 --- a/src/blockdata/transaction.rs +++ b/src/blockdata/transaction.rs @@ -86,7 +86,7 @@ impl OutPoint { /// let tx = &block.txdata[0]; /// /// // Coinbase transactions don't have any previous output. - /// assert_eq!(tx.input[0].previous_output.is_null(), true); + /// assert!(tx.input[0].previous_output.is_null()); /// ``` #[inline] pub fn is_null(&self) -> bool { From f02b3a84728bc913da6d5b0c1898d0fc8faa2dc5 Mon Sep 17 00:00:00 2001 From: Tobin Harding Date: Wed, 9 Mar 2022 12:49:41 +1100 Subject: [PATCH 3/4] Add code comment for emtpy input The line of code `let mut have_witness = self.input.is_empty();` is puzzling if one does not know _why_ we serialize in BIP141 style when there are no inputs. Add a code comment to save devs spending time trying to work out _why_ this is correct. --- src/blockdata/transaction.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/blockdata/transaction.rs b/src/blockdata/transaction.rs index fcb3f0224..e3c9b1f73 100644 --- a/src/blockdata/transaction.rs +++ b/src/blockdata/transaction.rs @@ -588,6 +588,8 @@ impl Encodable for Transaction { ) -> Result { let mut len = 0; len += self.version.consensus_encode(&mut s)?; + // To avoid serialization ambiguity, no inputs means we use BIP141 serialization (see + // `Transaction` docs for full explanation). let mut have_witness = self.input.is_empty(); for input in &self.input { if !input.witness.is_empty() { From e503f1433104b6481ac686d40c0f874e555f3ade Mon Sep 17 00:00:00 2001 From: Tobin Harding Date: Wed, 9 Mar 2022 13:04:09 +1100 Subject: [PATCH 4/4] Improve docs: blockdata::transaction Improve the rustdocs for the `blockdata::transaction` module: - Use full sentences (capitalisation and full stop) - Use third person tense instead of imperative - Improve wording/grammar - Use backticks in links - Use 100 character column width if it improves readability Nothing too controversial here :) --- src/blockdata/transaction.rs | 60 +++++++++++++++++++----------------- 1 file changed, 31 insertions(+), 29 deletions(-) diff --git a/src/blockdata/transaction.rs b/src/blockdata/transaction.rs index e3c9b1f73..0f7a1624e 100644 --- a/src/blockdata/transaction.rs +++ b/src/blockdata/transaction.rs @@ -42,18 +42,21 @@ use consensus::encode::MAX_VEC_SIZE; use hash_types::{SigHash, Txid, Wtxid}; use VarInt; -/// A reference to a transaction output +#[cfg(doc)] +use util::sighash::SchnorrSigHashType; + +/// A reference to a transaction output. #[derive(Copy, Clone, Debug, Eq, Hash, PartialEq, PartialOrd, Ord)] pub struct OutPoint { - /// The referenced transaction's txid + /// The referenced transaction's txid. pub txid: Txid, - /// The index of the referenced output in its transaction's vout + /// The index of the referenced output in its transaction's vout. pub vout: u32, } serde_struct_human_string_impl!(OutPoint, "an OutPoint", txid, vout); impl OutPoint { - /// Create a new [OutPoint]. + /// Creates a new [`OutPoint`]. #[inline] pub fn new(txid: Txid, vout: u32) -> OutPoint { OutPoint { @@ -64,8 +67,7 @@ impl OutPoint { /// Creates a "null" `OutPoint`. /// - /// This value is used for coinbase transactions because they don't have - /// any previous outputs. + /// This value is used for coinbase transactions because they don't have any previous outputs. #[inline] pub fn null() -> OutPoint { OutPoint { @@ -146,7 +148,7 @@ impl error::Error for ParseOutPointError { } /// Parses a string-encoded transaction index (vout). -/// It does not permit leading zeroes or non-digit characters. +/// Does not permit leading zeroes or non-digit characters. fn parse_vout(s: &str) -> Result { if s.len() > 1 { let first = s.chars().next().unwrap(); @@ -183,10 +185,10 @@ impl ::core::str::FromStr for OutPoint { #[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Debug, Hash)] #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] pub struct TxIn { - /// The reference to the previous output that is being used an an input + /// The reference to the previous output that is being used an an input. pub previous_output: OutPoint, /// The script which pushes values on the stack which will cause - /// the referenced output's script to accept + /// the referenced output's script to be accepted. pub script_sig: Script, /// The sequence number, which suggests to miners which of two /// conflicting transactions should be preferred, or 0xFFFFFFFF @@ -216,13 +218,13 @@ impl Default for TxIn { #[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Debug, Hash)] #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] pub struct TxOut { - /// The value of the output, in satoshis + /// The value of the output, in satoshis. pub value: u64, - /// The script which must satisfy for the output to be spent + /// The script which must be satisfied for the output to be spent. pub script_pubkey: Script } -// This is used as a "null txout" in consensus signing code +// This is used as a "null txout" in consensus signing code. impl Default for TxOut { fn default() -> TxOut { TxOut { value: 0xffffffffffffffff, script_pubkey: Script::new() } @@ -264,12 +266,11 @@ impl Default for TxOut { pub struct Transaction { /// The protocol version, is currently expected to be 1 or 2 (BIP 68). pub version: i32, - /// Block number before which this transaction is valid, or 0 for - /// valid immediately. + /// Block number before which this transaction is valid, or 0 for valid immediately. pub lock_time: u32, - /// List of inputs + /// List of transaction inputs. pub input: Vec, - /// List of outputs + /// List of transaction outputs. pub output: Vec, } @@ -499,7 +500,7 @@ impl Transaction { } } - /// Shorthand for [Self::verify_with_flags] with flag [bitcoinconsensus::VERIFY_ALL] + /// Shorthand for [`Self::verify_with_flags`] with flag [`bitcoinconsensus::VERIFY_ALL`]. #[cfg(feature="bitcoinconsensus")] #[cfg_attr(docsrs, doc(cfg(feature = "bitcoinconsensus")))] pub fn verify(&self, spent: S) -> Result<(), script::Error> @@ -507,8 +508,8 @@ impl Transaction { self.verify_with_flags(spent, ::bitcoinconsensus::VERIFY_ALL) } - /// Verify that this transaction is able to spend its inputs - /// The lambda spent should not return the same TxOut twice! + /// Verify that this transaction is able to spend its inputs. + /// The `spent` closure should not return the same [`TxOut`] twice! #[cfg(feature="bitcoinconsensus")] #[cfg_attr(docsrs, doc(cfg(feature = "bitcoinconsensus")))] pub fn verify_with_flags(&self, mut spent: S, flags: F) -> Result<(), script::Error> @@ -673,29 +674,30 @@ impl fmt::Display for NonStandardSigHashType { #[cfg_attr(docsrs, doc(cfg(feature = "std")))] impl error::Error for NonStandardSigHashType {} -/// Legacy Hashtype of an input's signature +/// Legacy Hashtype of an input's signature. #[deprecated(since="0.28.0", note="Please use [`EcdsaSigHashType`] instead")] pub type SigHashType = EcdsaSigHashType; -/// Hashtype of an input's signature, encoded in the last byte of the signature -/// Fixed values so they can be casted as integer types for encoding -/// See also [`crate::SchnorrSigHashType`] +/// Hashtype of an input's signature, encoded in the last byte of the signature. +/// +/// Fixed values so they can be cast as integer types for encoding (see also +/// [`SchnorrSigHashType`]). #[derive(PartialEq, Eq, Debug, Copy, Clone)] pub enum EcdsaSigHashType { - /// 0x1: Sign all outputs + /// 0x1: Sign all outputs. All = 0x01, - /// 0x2: Sign no outputs --- anyone can choose the destination + /// 0x2: Sign no outputs --- anyone can choose the destination. None = 0x02, /// 0x3: Sign the output whose index matches this input's index. If none exists, /// sign the hash `0000000000000000000000000000000000000000000000000000000000000001`. /// (This rule is probably an unintentional C++ism, but it's consensus so we have /// to follow it.) Single = 0x03, - /// 0x81: Sign all outputs but only this input + /// 0x81: Sign all outputs but only this input. AllPlusAnyoneCanPay = 0x81, - /// 0x82: Sign no outputs and only this input + /// 0x82: Sign no outputs and only this input. NonePlusAnyoneCanPay = 0x82, - /// 0x83: Sign one output and only this input (see `Single` for what "one output" means) + /// 0x83: Sign one output and only this input (see `Single` for what "one output" means). SinglePlusAnyoneCanPay = 0x83 } serde_string_impl!(EcdsaSigHashType, "a EcdsaSigHashType data"); @@ -731,7 +733,7 @@ impl str::FromStr for EcdsaSigHashType { } impl EcdsaSigHashType { - /// Break the sighash flag into the "real" sighash flag and the ANYONECANPAY boolean + /// Splits the sighash flag into the "real" sighash flag and the ANYONECANPAY boolean. pub(crate) fn split_anyonecanpay_flag(self) -> (EcdsaSigHashType, bool) { match self { EcdsaSigHashType::All => (EcdsaSigHashType::All, false),