@@ -27,8 +27,14 @@ use miniscript::limits::{HEIGHT_TIME_THRESHOLD, SEQUENCE_LOCKTIME_TYPE_FLAG};
27
27
use miniscript:: types:: extra_props:: TimeLockInfo ;
28
28
#[ cfg( feature = "compiler" ) ]
29
29
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 ,
32
38
} ;
33
39
use { Error , ForEach , ForEachKey , MiniscriptKey } ;
34
40
@@ -126,27 +132,115 @@ impl fmt::Display for PolicyError {
126
132
}
127
133
128
134
impl < Pk : MiniscriptKey > Policy < Pk > {
129
- /// Single-Node compilation
135
+ /// Create a Huffman Tree from compiled [Miniscript] nodes
130
136
#[ 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 ( ) )
134
166
}
135
167
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)
137
171
#[ 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." ) ) ,
142
229
}
143
230
}
144
231
145
232
/// Compile the [`Tr`] descriptor into optimized [`TapTree`] implementation
233
+ // TODO: We might require other compile errors for Taproot. Will discuss and update.
146
234
#[ cfg( feature = "compiler" ) ]
147
235
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
+ ) ?;
150
244
Ok ( tree)
151
245
}
152
246
@@ -250,6 +344,30 @@ impl<Pk: MiniscriptKey> Policy<Pk> {
250
344
}
251
345
}
252
346
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
+
253
371
/// Get all keys in the policy
254
372
pub fn keys ( & self ) -> Vec < & Pk > {
255
373
match * self {
0 commit comments