Skip to content

Commit 1a3605d

Browse files
committed
Merge #733: Remove TranslatePk trait and clean up Translator trait
36b0659 translator: make target pk and error associated types (Andrew Poelstra) c330d0b remove the TranslatePk trait (Andrew Poelstra) 6e1d6bb descriptor: stop using TranslatePk in mod.rs (Andrew Poelstra) b80296b descriptor: stop using TranslatePk in tr.rs (Andrew Poelstra) f89cf25 descriptor: stop using TranslatePk in sh.rs (Andrew Poelstra) 4654b98 descriptor: stop using TranslatePk in segwitv0.rs (Andrew Poelstra) 9c91586 descriptor: stop using TranslatePk in bare.rs (Andrew Poelstra) e73a141 miniscript: stop using TranslatePk (Andrew Poelstra) Pull request description: This may be too much of a breaking change and an instance of rust-bitcoin/rust-bitcoin#3166, but the existing traits are *really* annoying to use. But this change is not essential to any of my other work so feel free to concept NACK it. There are two changes here: * This drops the `TranslatePk` trait which had no business existing. It attempts to be generic over "things whose keys can be translated", but is missing impls (on keys themselves, for example) and anyway it is basically impossible to usefully be generic over this trait since it has an associated `Output` type which is unconstrained except by a comment saying "this must be `Self<Q>`. Really, the only purpose of this trait is to force users to write `use miniscript::TranslatePk` every time they want access to the `translate_pk` method. * It moves two of the generics in `Translator` from generics to associated types. This makes it far more ergonomic to implement and require the trait, since you no longer need to write 3 separate generics everywhere when two are implied. (Actually, all three are implied, including the source `Pk` type, but in practice users want to constrain this to match an existing key type that they've got. So I left it as a generic rather than an associated type.) ACKs for top commit: sanket1729: ACK 36b0659 Tree-SHA512: 0f7989925af2f9857146eb00e41a81ec1bbaf4c94b0003230835d3979ccb7913d8c958f1293f501e36ddcbaf81e8a88ada261371d9b79c4c85fec3ee195ffc07
2 parents 409374a + 36b0659 commit 1a3605d

File tree

18 files changed

+278
-285
lines changed

18 files changed

+278
-285
lines changed

bitcoind-tests/tests/setup/test_util.rs

+19-13
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ use bitcoin::secp256k1;
2626
use miniscript::descriptor::{SinglePub, SinglePubKey};
2727
use miniscript::{
2828
bitcoin, hash256, Descriptor, DescriptorPublicKey, Error, Miniscript, ScriptContext,
29-
TranslatePk, Translator,
29+
Translator,
3030
};
3131
use rand::RngCore;
3232
use secp256k1::XOnlyPublicKey;
@@ -155,8 +155,11 @@ pub fn parse_insane_ms<Ctx: ScriptContext>(
155155
#[derive(Debug, Clone)]
156156
struct StrDescPubKeyTranslator<'a>(usize, &'a PubData);
157157

158-
impl<'a> Translator<String, DescriptorPublicKey, ()> for StrDescPubKeyTranslator<'a> {
159-
fn pk(&mut self, pk_str: &String) -> Result<DescriptorPublicKey, ()> {
158+
impl<'a> Translator<String> for StrDescPubKeyTranslator<'a> {
159+
type TargetPk = DescriptorPublicKey;
160+
type Error = core::convert::Infallible;
161+
162+
fn pk(&mut self, pk_str: &String) -> Result<Self::TargetPk, Self::Error> {
160163
let avail = !pk_str.ends_with('!');
161164
if avail {
162165
self.0 += 1;
@@ -181,22 +184,22 @@ impl<'a> Translator<String, DescriptorPublicKey, ()> for StrDescPubKeyTranslator
181184
}
182185
}
183186

184-
fn sha256(&mut self, sha256: &String) -> Result<sha256::Hash, ()> {
187+
fn sha256(&mut self, sha256: &String) -> Result<sha256::Hash, Self::Error> {
185188
let sha = sha256::Hash::from_str(sha256).unwrap();
186189
Ok(sha)
187190
}
188191

189-
fn hash256(&mut self, hash256: &String) -> Result<hash256::Hash, ()> {
192+
fn hash256(&mut self, hash256: &String) -> Result<hash256::Hash, Self::Error> {
190193
let hash256 = hash256::Hash::from_str(hash256).unwrap();
191194
Ok(hash256)
192195
}
193196

194-
fn ripemd160(&mut self, ripemd160: &String) -> Result<ripemd160::Hash, ()> {
197+
fn ripemd160(&mut self, ripemd160: &String) -> Result<ripemd160::Hash, Self::Error> {
195198
let ripemd160 = ripemd160::Hash::from_str(ripemd160).unwrap();
196199
Ok(ripemd160)
197200
}
198201

199-
fn hash160(&mut self, hash160: &String) -> Result<hash160::Hash, ()> {
202+
fn hash160(&mut self, hash160: &String) -> Result<hash160::Hash, Self::Error> {
200203
let hash160 = hash160::Hash::from_str(hash160).unwrap();
201204
Ok(hash160)
202205
}
@@ -208,8 +211,11 @@ impl<'a> Translator<String, DescriptorPublicKey, ()> for StrDescPubKeyTranslator
208211
#[derive(Debug, Clone)]
209212
struct StrTranslatorLoose<'a>(usize, &'a PubData);
210213

211-
impl<'a> Translator<String, DescriptorPublicKey, ()> for StrTranslatorLoose<'a> {
212-
fn pk(&mut self, pk_str: &String) -> Result<DescriptorPublicKey, ()> {
214+
impl<'a> Translator<String> for StrTranslatorLoose<'a> {
215+
type TargetPk = DescriptorPublicKey;
216+
type Error = core::convert::Infallible;
217+
218+
fn pk(&mut self, pk_str: &String) -> Result<Self::TargetPk, Self::Error> {
213219
let avail = !pk_str.ends_with('!');
214220
if avail {
215221
self.0 += 1;
@@ -238,22 +244,22 @@ impl<'a> Translator<String, DescriptorPublicKey, ()> for StrTranslatorLoose<'a>
238244
}
239245
}
240246

241-
fn sha256(&mut self, sha256: &String) -> Result<sha256::Hash, ()> {
247+
fn sha256(&mut self, sha256: &String) -> Result<sha256::Hash, Self::Error> {
242248
let sha = sha256::Hash::from_str(sha256).unwrap();
243249
Ok(sha)
244250
}
245251

246-
fn hash256(&mut self, hash256: &String) -> Result<hash256::Hash, ()> {
252+
fn hash256(&mut self, hash256: &String) -> Result<hash256::Hash, Self::Error> {
247253
let hash256 = hash256::Hash::from_str(hash256).unwrap();
248254
Ok(hash256)
249255
}
250256

251-
fn ripemd160(&mut self, ripemd160: &String) -> Result<ripemd160::Hash, ()> {
257+
fn ripemd160(&mut self, ripemd160: &String) -> Result<ripemd160::Hash, Self::Error> {
252258
let ripemd160 = ripemd160::Hash::from_str(ripemd160).unwrap();
253259
Ok(ripemd160)
254260
}
255261

256-
fn hash160(&mut self, hash160: &String) -> Result<hash160::Hash, ()> {
262+
fn hash160(&mut self, hash160: &String) -> Result<hash160::Hash, Self::Error> {
257263
let hash160 = hash160::Hash::from_str(hash160).unwrap();
258264
Ok(hash160)
259265
}

examples/big.rs

+7-4
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ use miniscript::policy::{Concrete, Liftable};
1818
use miniscript::psbt::PsbtExt;
1919
use miniscript::{
2020
translate_hash_fail, DefiniteDescriptorKey, Descriptor, DescriptorPublicKey, MiniscriptKey,
21-
TranslatePk, Translator,
21+
Translator,
2222
};
2323
use secp256k1::Secp256k1;
2424
fn main() {
@@ -82,12 +82,15 @@ struct StrPkTranslator {
8282
pk_map: HashMap<String, XOnlyPublicKey>,
8383
}
8484

85-
impl Translator<String, XOnlyPublicKey, ()> for StrPkTranslator {
86-
fn pk(&mut self, pk: &String) -> Result<XOnlyPublicKey, ()> {
85+
impl Translator<String> for StrPkTranslator {
86+
type TargetPk = XOnlyPublicKey;
87+
type Error = ();
88+
89+
fn pk(&mut self, pk: &String) -> Result<XOnlyPublicKey, Self::Error> {
8790
self.pk_map.get(pk).copied().ok_or(())
8891
}
8992

9093
// We don't need to implement these methods as we are not using them in the policy.
9194
// Fail if we encounter any hash fragments. See also translate_hash_clone! macro.
92-
translate_hash_fail!(String, XOnlyPublicKey, ());
95+
translate_hash_fail!(String, XOnlyPublicKey, Self::Error);
9396
}

examples/taproot.rs

+7-4
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use miniscript::bitcoin::secp256k1::rand;
88
use miniscript::bitcoin::{Network, WitnessVersion};
99
use miniscript::descriptor::DescriptorType;
1010
use miniscript::policy::Concrete;
11-
use miniscript::{translate_hash_fail, Descriptor, Miniscript, Tap, TranslatePk, Translator};
11+
use miniscript::{translate_hash_fail, Descriptor, Miniscript, Tap, Translator};
1212

1313
// Refer to https://github.com/sanket1729/adv_btc_workshop/blob/master/workshop.md#creating-a-taproot-descriptor
1414
// for a detailed explanation of the policy and it's compilation
@@ -17,14 +17,17 @@ struct StrPkTranslator {
1717
pk_map: HashMap<String, XOnlyPublicKey>,
1818
}
1919

20-
impl Translator<String, XOnlyPublicKey, ()> for StrPkTranslator {
21-
fn pk(&mut self, pk: &String) -> Result<XOnlyPublicKey, ()> {
20+
impl Translator<String> for StrPkTranslator {
21+
type TargetPk = XOnlyPublicKey;
22+
type Error = ();
23+
24+
fn pk(&mut self, pk: &String) -> Result<XOnlyPublicKey, Self::Error> {
2225
self.pk_map.get(pk).copied().ok_or(())
2326
}
2427

2528
// We don't need to implement these methods as we are not using them in the policy.
2629
// Fail if we encounter any hash fragments. See also translate_hash_clone! macro.
27-
translate_hash_fail!(String, XOnlyPublicKey, ());
30+
translate_hash_fail!(String, XOnlyPublicKey, Self::Error);
2831
}
2932

3033
fn main() {

src/descriptor/bare.rs

+21-35
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ use crate::prelude::*;
2323
use crate::util::{varint_len, witness_to_scriptsig};
2424
use crate::{
2525
BareCtx, Error, ForEachKey, FromStrKey, Miniscript, MiniscriptKey, Satisfier, ToPublicKey,
26-
TranslateErr, TranslatePk, Translator,
26+
TranslateErr, Translator,
2727
};
2828

2929
/// Create a Bare Descriptor. That is descriptor that is
@@ -92,6 +92,14 @@ impl<Pk: MiniscriptKey> Bare<Pk> {
9292
let scriptsig_len = self.ms.max_satisfaction_size()?;
9393
Ok(4 * (varint_len(scriptsig_len) + scriptsig_len))
9494
}
95+
96+
/// Converts the keys in the script from one type to another.
97+
pub fn translate_pk<T>(&self, t: &mut T) -> Result<Bare<T::TargetPk>, TranslateErr<T::Error>>
98+
where
99+
T: Translator<Pk>,
100+
{
101+
Bare::new(self.ms.translate_pk(t)?).map_err(TranslateErr::OuterError)
102+
}
95103
}
96104

97105
impl<Pk: MiniscriptKey + ToPublicKey> Bare<Pk> {
@@ -190,21 +198,6 @@ impl<Pk: MiniscriptKey> ForEachKey<Pk> for Bare<Pk> {
190198
}
191199
}
192200

193-
impl<P, Q> TranslatePk<P, Q> for Bare<P>
194-
where
195-
P: MiniscriptKey,
196-
Q: MiniscriptKey,
197-
{
198-
type Output = Bare<Q>;
199-
200-
fn translate_pk<T, E>(&self, t: &mut T) -> Result<Bare<Q>, TranslateErr<E>>
201-
where
202-
T: Translator<P, Q, E>,
203-
{
204-
Bare::new(self.ms.translate_pk(t)?).map_err(TranslateErr::OuterError)
205-
}
206-
}
207-
208201
/// A bare PkH descriptor at top level
209202
#[derive(Clone, Ord, PartialOrd, Eq, PartialEq, Hash)]
210203
pub struct Pkh<Pk: MiniscriptKey> {
@@ -260,6 +253,18 @@ impl<Pk: MiniscriptKey> Pkh<Pk> {
260253
note = "Use max_weight_to_satisfy instead. The method to count bytes was redesigned and the results will differ from max_weight_to_satisfy. For more details check rust-bitcoin/rust-miniscript#476."
261254
)]
262255
pub fn max_satisfaction_weight(&self) -> usize { 4 * (1 + 73 + BareCtx::pk_len(&self.pk)) }
256+
257+
/// Converts the keys in a script from one type to another.
258+
pub fn translate_pk<T>(&self, t: &mut T) -> Result<Pkh<T::TargetPk>, TranslateErr<T::Error>>
259+
where
260+
T: Translator<Pk>,
261+
{
262+
let res = Pkh::new(t.pk(&self.pk)?);
263+
match res {
264+
Ok(pk) => Ok(pk),
265+
Err(e) => Err(TranslateErr::OuterError(Error::from(e))),
266+
}
267+
}
263268
}
264269

265270
impl<Pk: MiniscriptKey + ToPublicKey> Pkh<Pk> {
@@ -391,22 +396,3 @@ impl<Pk: FromStrKey> core::str::FromStr for Pkh<Pk> {
391396
impl<Pk: MiniscriptKey> ForEachKey<Pk> for Pkh<Pk> {
392397
fn for_each_key<'a, F: FnMut(&'a Pk) -> bool>(&'a self, mut pred: F) -> bool { pred(&self.pk) }
393398
}
394-
395-
impl<P, Q> TranslatePk<P, Q> for Pkh<P>
396-
where
397-
P: MiniscriptKey,
398-
Q: MiniscriptKey,
399-
{
400-
type Output = Pkh<Q>;
401-
402-
fn translate_pk<T, E>(&self, t: &mut T) -> Result<Self::Output, TranslateErr<E>>
403-
where
404-
T: Translator<P, Q, E>,
405-
{
406-
let res = Pkh::new(t.pk(&self.pk)?);
407-
match res {
408-
Ok(pk) => Ok(pk),
409-
Err(e) => Err(TranslateErr::OuterError(Error::from(e))),
410-
}
411-
}
412-
}

src/descriptor/key.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -1003,9 +1003,7 @@ impl DefiniteDescriptorKey {
10031003
/// always return a compressed key
10041004
///
10051005
/// Will return an error if the descriptor key has any hardened derivation steps in its path. To
1006-
/// avoid this error you should replace any such public keys first with [`translate_pk`].
1007-
///
1008-
/// [`translate_pk`]: crate::TranslatePk::translate_pk
1006+
/// avoid this error you should replace any such public keys first with [`crate::Descriptor::translate_pk`].
10091007
pub fn derive_public_key<C: Verification>(
10101008
&self,
10111009
secp: &Secp256k1<C>,

0 commit comments

Comments
 (0)