Define transaction::TransactionData::map_authorization

This commit is contained in:
Kris Nuttycombe 2022-02-03 13:45:41 -07:00
parent caba2dc8b0
commit af9edcea25
6 changed files with 128 additions and 0 deletions

View File

@ -76,6 +76,16 @@ pub struct Witness<T> {
pub payload: T, pub payload: T,
} }
impl<T> Witness<T> {
pub fn map_payload<U, F: FnOnce(T) -> U>(self, f: F) -> Witness<U> {
Witness {
extension_id: self.extension_id,
mode: self.mode,
payload: f(self.payload),
}
}
}
impl Witness<AuthData> { impl Witness<AuthData> {
/// Produce the intermediate format for an extension-specific witness /// Produce the intermediate format for an extension-specific witness
/// type. /// type.

View File

@ -28,6 +28,11 @@ impl Authorization for Unauthorized {
type SpendAuth = (); type SpendAuth = ();
} }
pub trait MapAuth<A: Authorization, B: Authorization> {
fn map_spend_auth(&self, s: A::SpendAuth) -> B::SpendAuth;
fn map_authorization(&self, a: A) -> B;
}
/// Reads an [`orchard::Bundle`] from a v5 transaction format. /// Reads an [`orchard::Bundle`] from a v5 transaction format.
pub fn read_v5_bundle<R: Read>( pub fn read_v5_bundle<R: Read>(
mut reader: R, mut reader: R,

View File

@ -47,6 +47,12 @@ impl Authorization for Authorized {
type AuthSig = redjubjub::Signature; type AuthSig = redjubjub::Signature;
} }
pub trait MapAuth<A: Authorization, B: Authorization> {
fn map_proof(&self, p: A::Proof) -> B::Proof;
fn map_auth_sig(&self, s: A::AuthSig) -> B::AuthSig;
fn map_authorization(&self, a: A) -> B;
}
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct Bundle<A: Authorization> { pub struct Bundle<A: Authorization> {
pub shielded_spends: Vec<SpendDescription<A>>, pub shielded_spends: Vec<SpendDescription<A>>,
@ -55,6 +61,39 @@ pub struct Bundle<A: Authorization> {
pub authorization: A, pub authorization: A,
} }
impl<A: Authorization> Bundle<A> {
pub fn map_authorization<B: Authorization, F: MapAuth<A, B>>(self, f: F) -> Bundle<B> {
Bundle {
shielded_spends: self
.shielded_spends
.into_iter()
.map(|d| SpendDescription {
cv: d.cv,
anchor: d.anchor,
nullifier: d.nullifier,
rk: d.rk,
zkproof: f.map_proof(d.zkproof),
spend_auth_sig: f.map_auth_sig(d.spend_auth_sig),
})
.collect(),
shielded_outputs: self
.shielded_outputs
.into_iter()
.map(|o| OutputDescription {
cv: o.cv,
cmu: o.cmu,
ephemeral_key: o.ephemeral_key,
enc_ciphertext: o.enc_ciphertext,
out_ciphertext: o.out_ciphertext,
zkproof: f.map_proof(o.zkproof),
})
.collect(),
value_balance: self.value_balance,
authorization: f.map_authorization(self.authorization),
}
}
}
#[derive(Clone)] #[derive(Clone)]
pub struct SpendDescription<A: Authorization> { pub struct SpendDescription<A: Authorization> {
pub cv: jubjub::ExtendedPoint, pub cv: jubjub::ExtendedPoint,

View File

@ -22,6 +22,11 @@ impl Authorization for Authorized {
type ScriptSig = Script; type ScriptSig = Script;
} }
pub trait MapAuth<A: Authorization, B: Authorization> {
fn map_script_sig(&self, s: A::ScriptSig) -> B::ScriptSig;
fn map_authorization(&self, s: A) -> B;
}
#[derive(Debug, Clone, PartialEq)] #[derive(Debug, Clone, PartialEq)]
pub struct Bundle<A: Authorization> { pub struct Bundle<A: Authorization> {
pub vin: Vec<TxIn<A>>, pub vin: Vec<TxIn<A>>,
@ -29,6 +34,24 @@ pub struct Bundle<A: Authorization> {
pub authorization: A, pub authorization: A,
} }
impl<A: Authorization> Bundle<A> {
pub fn map_authorization<B: Authorization, F: MapAuth<A, B>>(self, f: F) -> Bundle<B> {
Bundle {
vin: self
.vin
.into_iter()
.map(|txin| TxIn {
prevout: txin.prevout,
script_sig: f.map_script_sig(txin.script_sig),
sequence: txin.sequence,
})
.collect(),
vout: self.vout,
authorization: f.map_authorization(self.authorization),
}
}
}
#[derive(Clone, Debug, PartialEq)] #[derive(Clone, Debug, PartialEq)]
pub struct OutPoint { pub struct OutPoint {
hash: [u8; 32], hash: [u8; 32],

View File

@ -28,6 +28,11 @@ impl Authorization for Authorized {
type Witness = tze::AuthData; type Witness = tze::AuthData;
} }
pub trait MapAuth<A: Authorization, B: Authorization> {
fn map_witness(&self, s: A::Witness) -> B::Witness;
fn map_authorization(&self, s: A) -> B;
}
#[derive(Debug, Clone, PartialEq)] #[derive(Debug, Clone, PartialEq)]
pub struct Bundle<A: Authorization> { pub struct Bundle<A: Authorization> {
pub vin: Vec<TzeIn<A::Witness>>, pub vin: Vec<TzeIn<A::Witness>>,
@ -35,6 +40,23 @@ pub struct Bundle<A: Authorization> {
pub authorization: A, pub authorization: A,
} }
impl<A: Authorization> Bundle<A> {
pub fn map_authorization<B: Authorization, F: MapAuth<A, B>>(self, f: F) -> Bundle<B> {
Bundle {
vin: self
.vin
.into_iter()
.map(|tzein| TzeIn {
prevout: tzein.prevout,
witness: tzein.witness.map_payload(|p| f.map_witness(p)),
})
.collect(),
vout: self.vout,
authorization: f.map_authorization(self.authorization),
}
}
}
#[derive(Clone, Debug, PartialEq)] #[derive(Clone, Debug, PartialEq)]
pub struct OutPoint { pub struct OutPoint {
txid: TxId, txid: TxId,

View File

@ -370,6 +370,35 @@ impl<A: Authorization> TransactionData<A> {
digester.digest_tze(self.tze_bundle.as_ref()), digester.digest_tze(self.tze_bundle.as_ref()),
) )
} }
pub fn map_authorization<B: Authorization>(
self,
f_transparent: impl transparent::MapAuth<A::TransparentAuth, B::TransparentAuth>,
f_sapling: impl sapling::MapAuth<A::SaplingAuth, B::SaplingAuth>,
mut f_orchard: impl orchard_serialization::MapAuth<A::OrchardAuth, B::OrchardAuth>,
#[cfg(feature = "zfuture")] f_tze: impl tze::MapAuth<A::TzeAuth, B::TzeAuth>,
) -> TransactionData<B> {
TransactionData {
version: self.version,
consensus_branch_id: self.consensus_branch_id,
lock_time: self.lock_time,
expiry_height: self.expiry_height,
transparent_bundle: self
.transparent_bundle
.map(|b| b.map_authorization(f_transparent)),
sprout_bundle: self.sprout_bundle,
sapling_bundle: self.sapling_bundle.map(|b| b.map_authorization(f_sapling)),
orchard_bundle: self.orchard_bundle.map(|b| {
b.authorize(
&mut f_orchard,
|f, _, s| f.map_spend_auth(s),
|f, a| f.map_authorization(a),
)
}),
#[cfg(feature = "zfuture")]
tze_bundle: self.tze_bundle.map(|b| b.map_authorization(f_tze)),
}
}
} }
impl<A: Authorization> std::fmt::Debug for TransactionData<A> { impl<A: Authorization> std::fmt::Debug for TransactionData<A> {