Skip to content

Commit e507066

Browse files
trivial P2Tr compiler done
Taproot internal-key extraction from policy done
1 parent b36c116 commit e507066

File tree

4 files changed

+145
-33
lines changed

4 files changed

+145
-33
lines changed

src/policy/compiler.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ type PolicyCache<Pk, Ctx> =
3838

3939
///Ordered f64 for comparison
4040
#[derive(Copy, Clone, PartialEq, PartialOrd, Debug)]
41-
struct OrdF64(f64);
41+
pub(crate) struct OrdF64(pub f64);
4242

4343
impl Eq for OrdF64 {}
4444
impl Ord for OrdF64 {

src/policy/concrete.rs

+131-13
Original file line numberDiff line numberDiff line change
@@ -27,8 +27,14 @@ use miniscript::limits::{HEIGHT_TIME_THRESHOLD, SEQUENCE_LOCKTIME_TYPE_FLAG};
2727
use miniscript::types::extra_props::TimeLockInfo;
2828
#[cfg(feature = "compiler")]
2929
use {
30-
descriptor::TapTree, miniscript::ScriptContext, policy::compiler,
31-
policy::compiler::CompilerError, std::sync::Arc, Descriptor, Miniscript, Tap,
30+
descriptor::TapTree,
31+
miniscript::ScriptContext,
32+
policy::compiler::{CompilerError, OrdF64},
33+
policy::{compiler, Liftable, Semantic},
34+
std::cmp::Reverse,
35+
std::collections::BinaryHeap,
36+
std::sync::Arc,
37+
Descriptor, Miniscript, Tap,
3238
};
3339
use {Error, ForEach, ForEachKey, MiniscriptKey};
3440

@@ -126,27 +132,115 @@ impl fmt::Display for PolicyError {
126132
}
127133

128134
impl<Pk: MiniscriptKey> Policy<Pk> {
129-
/// Single-Node compilation
135+
/// Create a Huffman Tree from compiled [Miniscript] nodes
130136
#[cfg(feature = "compiler")]
131-
fn compile_leaf_taptree(&self) -> Result<TapTree<Pk>, Error> {
132-
let compilation = self.compile::<Tap>().unwrap();
133-
Ok(TapTree::Leaf(Arc::new(compilation)))
137+
fn with_huffman_tree<T>(
138+
ms: Vec<(OrdF64, Miniscript<Pk, Tap>)>,
139+
f: T,
140+
) -> Result<TapTree<Pk>, Error>
141+
where
142+
T: Fn(OrdF64) -> OrdF64,
143+
{
144+
// Pattern match terminal Or/ Terminal (with equal odds)
145+
let mut node_weights = BinaryHeap::<(Reverse<OrdF64>, Arc<TapTree<Pk>>)>::new();
146+
for (prob, script) in ms {
147+
node_weights.push((Reverse(f(prob)), Arc::from(TapTree::Leaf(Arc::new(script)))));
148+
}
149+
if node_weights.is_empty() {
150+
return Err(errstr("Empty Miniscript compilation"));
151+
}
152+
while node_weights.len() > 1 {
153+
let (p1, s1) = node_weights.pop().expect("len must atleast be two");
154+
let (p2, s2) = node_weights.pop().expect("len must atleast be two");
155+
156+
let p = (p1.0).0 + (p2.0).0;
157+
node_weights.push((Reverse(OrdF64(p)), Arc::from(TapTree::Tree(s1, s2))));
158+
}
159+
160+
debug_assert!(node_weights.len() == 1);
161+
let node = node_weights
162+
.pop()
163+
.expect("huffman tree algorithm is broken")
164+
.1;
165+
Ok((*node).clone())
134166
}
135167

136-
/// Extract the Taproot internal_key from policy tree.
168+
/// Flatten the [`Policy`] tree structure into a Vector with corresponding leaf probability
169+
// TODO: 1. Can try to push the maximum of scaling factors and accordingly update later for
170+
// TODO: 1. integer metric. (Accordingly change metrics everywhere)
137171
#[cfg(feature = "compiler")]
138-
fn extract_key(&self, unspendable_key: Option<Pk>) -> Result<(Pk, &Policy<Pk>), Error> {
139-
match unspendable_key {
140-
Some(key) => Ok((key, self)),
141-
None => Err(errstr("No internal key found")),
172+
fn to_tapleaf_prob_vec(&self, prob: f64) -> Vec<(f64, Policy<Pk>)> {
173+
match *self {
174+
Policy::Or(ref subs) => {
175+
let total_odds: usize = subs.iter().map(|(ref k, _)| k).sum();
176+
subs.iter()
177+
.map(|(k, ref policy)| {
178+
policy.to_tapleaf_prob_vec(prob * *k as f64 / total_odds as f64)
179+
})
180+
.flatten()
181+
.collect::<Vec<_>>()
182+
}
183+
Policy::Threshold(k, ref subs) if k == 1 => {
184+
let total_odds = subs.len();
185+
subs.iter()
186+
.map(|policy| policy.to_tapleaf_prob_vec(prob / total_odds as f64))
187+
.flatten()
188+
.collect::<Vec<_>>()
189+
}
190+
ref x => vec![(prob, x.clone())],
191+
}
192+
}
193+
194+
/// Compile [`Policy::Or`] and [`Policy::Threshold`] according to odds
195+
#[cfg(feature = "compiler")]
196+
fn compile_tr_policy(&self) -> Result<TapTree<Pk>, Error> {
197+
let leaf_compilations: Vec<_> = self
198+
.to_tapleaf_prob_vec(1.0)
199+
.into_iter()
200+
.map(|(prob, ref policy)| (OrdF64(prob), policy.compile::<Tap>().unwrap()))
201+
.collect();
202+
let taptree = Self::with_huffman_tree(leaf_compilations, |x| x).unwrap();
203+
Ok(taptree)
204+
}
205+
206+
/// Extract the internal_key from policy tree.
207+
#[cfg(feature = "compiler")]
208+
fn extract_key(self, unspendable_key: Option<Pk>) -> Result<(Pk, Policy<Pk>), Error> {
209+
// Making sure the borrow ends before you move the value.
210+
let mut internal_key: Option<Pk> = None;
211+
{
212+
let semantic_policy = (&self).lift()?;
213+
let concrete_keys = (&self).keys().into_iter().collect::<HashSet<_>>();
214+
for key in concrete_keys.iter() {
215+
if semantic_policy
216+
.clone()
217+
.satisfy_constraint(&Semantic::KeyHash(key.to_pubkeyhash()), true)
218+
== Semantic::Trivial
219+
{
220+
internal_key = Some((*key).clone());
221+
break;
222+
}
223+
}
224+
}
225+
match (internal_key, unspendable_key) {
226+
(Some(ref key), _) => Ok((key.clone(), self.translate_unsatisfiable_pk(&key))),
227+
(_, Some(key)) => Ok((key, self)),
228+
_ => Err(errstr("No viable internal key found.")),
142229
}
143230
}
144231

145232
/// Compile the [`Tr`] descriptor into optimized [`TapTree`] implementation
233+
// TODO: We might require other compile errors for Taproot. Will discuss and update.
146234
#[cfg(feature = "compiler")]
147235
pub fn compile_tr(&self, unspendable_key: Option<Pk>) -> Result<Descriptor<Pk>, Error> {
148-
let (internal_key, policy) = self.extract_key(unspendable_key)?;
149-
let tree = Descriptor::new_tr(internal_key, Some(policy.compile_leaf_taptree()?))?;
236+
let (internal_key, policy) = self.clone().extract_key(unspendable_key)?;
237+
let tree = Descriptor::new_tr(
238+
internal_key,
239+
match policy {
240+
Policy::Trivial => None,
241+
policy => Some(policy.compile_tr_policy()?),
242+
},
243+
)?;
150244
Ok(tree)
151245
}
152246

@@ -250,6 +344,30 @@ impl<Pk: MiniscriptKey> Policy<Pk> {
250344
}
251345
}
252346

347+
/// Translate `Semantic::Key(key)` to `Semantic::Unsatisfiable` when extracting TapKey
348+
pub fn translate_unsatisfiable_pk(self, key: &Pk) -> Policy<Pk> {
349+
match self {
350+
Policy::Key(ref k) if k.clone() == *key => Policy::Unsatisfiable,
351+
Policy::And(subs) => Policy::And(
352+
subs.into_iter()
353+
.map(|sub| sub.translate_unsatisfiable_pk(key))
354+
.collect::<Vec<_>>(),
355+
),
356+
Policy::Or(subs) => Policy::Or(
357+
subs.into_iter()
358+
.map(|(k, sub)| (k, sub.translate_unsatisfiable_pk(key)))
359+
.collect::<Vec<_>>(),
360+
),
361+
Policy::Threshold(k, subs) => Policy::Threshold(
362+
k,
363+
subs.into_iter()
364+
.map(|sub| sub.translate_unsatisfiable_pk(key))
365+
.collect::<Vec<_>>(),
366+
),
367+
x => x,
368+
}
369+
}
370+
253371
/// Get all keys in the policy
254372
pub fn keys(&self) -> Vec<&Pk> {
255373
match *self {

src/policy/mod.rs

+7-10
Original file line numberDiff line numberDiff line change
@@ -374,16 +374,13 @@ mod tests {
374374
#[test]
375375
#[cfg(feature = "compiler")]
376376
fn single_leaf_tr_compile() {
377-
for k in 1..5 {
378-
let unspendable_key: String = "z".to_string();
379-
let policy: Concrete<String> = policy_str!("thresh({},pk(A),pk(B),pk(C),pk(D))", k);
380-
let descriptor = policy.compile_tr(Some(unspendable_key.clone())).unwrap();
377+
let unspendable_key: String = "UNSPENDABLE".to_string();
378+
let policy: Concrete<String> = policy_str!("thresh(2,pk(A),pk(A),pk(C),pk(D))");
379+
let descriptor = policy.compile_tr(Some(unspendable_key.clone())).unwrap();
381380

382-
let ms_compilation: Miniscript<String, Tap> = ms_str!("multi_a({},A,B,C,D)", k);
383-
let tree: TapTree<String> = TapTree::Leaf(Arc::new(ms_compilation));
384-
let expected_descriptor = Descriptor::new_tr(unspendable_key, Some(tree)).unwrap();
385-
386-
assert_eq!(descriptor, expected_descriptor);
387-
}
381+
let ms_compilation: Miniscript<String, Tap> = ms_str!("multi_a(2,C,D)");
382+
let tree: TapTree<String> = TapTree::Leaf(Arc::new(ms_compilation));
383+
let expected_descriptor = Descriptor::new_tr("A".to_string(), Some(tree)).unwrap();
384+
assert_eq!(descriptor, expected_descriptor);
388385
}
389386
}

src/policy/semantic.rs

+6-9
Original file line numberDiff line numberDiff line change
@@ -189,7 +189,7 @@ impl<Pk: MiniscriptKey> Policy<Pk> {
189189
// policy.
190190
// Witness is currently encoded as policy. Only accepts leaf fragment and
191191
// a normalized policy
192-
fn satisfy_constraint(self, witness: &Policy<Pk>, available: bool) -> Policy<Pk> {
192+
pub(crate) fn satisfy_constraint(self, witness: &Policy<Pk>, available: bool) -> Policy<Pk> {
193193
debug_assert!(self.clone().normalized() == self.clone());
194194
match *witness {
195195
// only for internal purposes, safe to use unreachable!
@@ -679,10 +679,7 @@ mod tests {
679679
let policy = StringPolicy::from_str("or(pkh(),older(1000))").unwrap();
680680
assert_eq!(
681681
policy,
682-
Policy::Threshold(
683-
1,
684-
vec![Policy::KeyHash("".to_owned()), Policy::Older(1000),]
685-
)
682+
Policy::Threshold(1, vec![Policy::KeyHash("".to_owned()), Policy::Older(1000)])
686683
);
687684
assert_eq!(policy.relative_timelocks(), vec![1000]);
688685
assert_eq!(policy.absolute_timelocks(), vec![]);
@@ -698,7 +695,7 @@ mod tests {
698695
policy,
699696
Policy::Threshold(
700697
1,
701-
vec![Policy::KeyHash("".to_owned()), Policy::Unsatisfiable,]
698+
vec![Policy::KeyHash("".to_owned()), Policy::Unsatisfiable],
702699
)
703700
);
704701
assert_eq!(policy.relative_timelocks(), vec![]);
@@ -711,7 +708,7 @@ mod tests {
711708
policy,
712709
Policy::Threshold(
713710
2,
714-
vec![Policy::KeyHash("".to_owned()), Policy::Unsatisfiable,]
711+
vec![Policy::KeyHash("".to_owned()), Policy::Unsatisfiable],
715712
)
716713
);
717714
assert_eq!(policy.relative_timelocks(), vec![]);
@@ -735,7 +732,7 @@ mod tests {
735732
Policy::Older(1000),
736733
Policy::Older(2000),
737734
Policy::Older(2000),
738-
]
735+
],
739736
)
740737
);
741738
assert_eq!(
@@ -759,7 +756,7 @@ mod tests {
759756
Policy::Older(1000),
760757
Policy::Unsatisfiable,
761758
Policy::Unsatisfiable,
762-
]
759+
],
763760
)
764761
);
765762
assert_eq!(

0 commit comments

Comments
 (0)