diff --git a/examples/exp_test.rs b/examples/exp_test.rs new file mode 100644 index 000000000..bf19a38fa --- /dev/null +++ b/examples/exp_test.rs @@ -0,0 +1,31 @@ +use miniscript::Descriptor; +use std::str::FromStr; + +fn test_one_descriptor(descriptor: &str) -> bool { + let res = Descriptor::::from_str(descriptor); + match res { + Ok(_) => true, + Err(err) => { + println!("{}", err); + false + } + } +} + +fn main() { + dbg!("inside test"); + assert!(test_one_descriptor("pk(musig(a))")); + // assert!(test_one_descriptor("pk(musig(a,b))")); + // assert!(test_one_descriptor("pk(musig(a,musig(b,c,d)))")); + // assert!(test_one_descriptor("sh(multi(2,musig(a,b),k1,k2))")); + // assert!(test_one_descriptor("pk(musig(musig(a,b),musig(c,d)))")); + // assert!(test_one_descriptor( + // "pk(musig(a,musig(b,c,musig(d,e,f,musig(g,h,i))),j))" + // )); + // assert!(test_one_descriptor( + // "sh(sortedmulti(3,k1,musig(a,b),musig(musig(c,d))))" + // )); + // Need to test, cases which can give errors + // assert!(!test_one_descriptor("pk(musig(a,b),musig(c))")); + // assert!(!test_one_descriptor("pk(k1,k2)")); +} \ No newline at end of file diff --git a/src/descriptor/bare.rs b/src/descriptor/bare.rs index 0b3159ab6..f7d65a232 100644 --- a/src/descriptor/bare.rs +++ b/src/descriptor/bare.rs @@ -147,6 +147,7 @@ impl Liftable for Bare { impl_from_tree!( Bare, fn from_tree(top: &expression::Tree) -> Result { + dbg!("from_tree in Bare"); let sub = Miniscript::::from_tree(top)?; BareCtx::top_level_checks(&sub)?; Bare::new(sub) diff --git a/src/descriptor/mod.rs b/src/descriptor/mod.rs index 46d1adfa9..546c9789a 100644 --- a/src/descriptor/mod.rs +++ b/src/descriptor/mod.rs @@ -718,6 +718,7 @@ impl_from_tree!( Descriptor, /// Parse an expression tree into a descriptor. fn from_tree(top: &expression::Tree) -> Result, Error> { + dbg!("from_tree of Descriptor"); Ok(match (top.name, top.args.len() as u32) { ("pkh", 1) => Descriptor::Pkh(Pkh::from_tree(top)?), ("wpkh", 1) => Descriptor::Wpkh(Wpkh::from_tree(top)?), @@ -736,6 +737,7 @@ impl_from_str!( // tr tree parsing has special code // Tr::from_str will check the checksum // match "tr(" to handle more extensibly + dbg!("from_str of Descriptor"); if s.starts_with("tr(") { Ok(Descriptor::Tr(Tr::from_str(s)?)) } else { @@ -817,6 +819,35 @@ mod tests { } } + fn test_one_descriptor(descriptor: &str) -> bool { + let res = Descriptor::::from_str(descriptor); + match res { + Ok(_) => true, + Err(err) => { + println!("{}", err); + false + } + } + } + + #[test] + fn test_musig_descriptors() { + dbg!("inside test"); + assert!(test_one_descriptor("pk(musig(a))")); + assert!(test_one_descriptor("pk(musig(a,b))")); + assert!(test_one_descriptor("pk(musig(a,musig(b,c,d)))")); + assert!(test_one_descriptor("sh(multi(2,musig(a,b),k1,k2))")); + assert!(test_one_descriptor("pk(musig(musig(a,b),musig(c,d)))")); + assert!(test_one_descriptor( + "pk(musig(a,musig(b,c,musig(d,e,f,musig(g,h,i))),j))" + )); + assert!(test_one_descriptor( + "sh(sortedmulti(3,k1,musig(a,b),musig(musig(c,d))))" + )); + // Need to test, cases which can give errors + assert!(!test_one_descriptor("pk(musig(a,b),musig(c))")); + assert!(!test_one_descriptor("pk(k1,k2)")); + } fn roundtrip_descriptor(s: &str) { let desc = Descriptor::::from_str(&s).unwrap(); let output = desc.to_string(); diff --git a/src/descriptor/sh.rs b/src/descriptor/sh.rs index d9202e67e..5ae30ee4f 100644 --- a/src/descriptor/sh.rs +++ b/src/descriptor/sh.rs @@ -95,6 +95,7 @@ impl_from_tree!( fn from_tree(top: &expression::Tree) -> Result { if top.name == "sh" && top.args.len() == 1 { let top = &top.args[0]; + dbg!("from_tree of sh"); let inner = match top.name { "wsh" => ShInner::Wsh(Wsh::from_tree(top)?), "wpkh" => ShInner::Wpkh(Wpkh::from_tree(top)?), diff --git a/src/descriptor/sortedmulti.rs b/src/descriptor/sortedmulti.rs index d3c21a239..bfd18bbae 100644 --- a/src/descriptor/sortedmulti.rs +++ b/src/descriptor/sortedmulti.rs @@ -74,6 +74,7 @@ impl SortedMultiVec { Pk: FromStr, ::Err: ToString, { + dbg!("from_tree of sorted multi vec"); if tree.args.is_empty() { return Err(errstr("no arguments given for sortedmulti")); } @@ -87,7 +88,7 @@ impl SortedMultiVec { .iter() .map(|sub| expression::terminal(sub, Pk::from_str)) .collect(); - + dbg!(&pks); pks.map(|pks| SortedMultiVec::new(k as usize, pks))? } diff --git a/src/expression.rs b/src/expression.rs index 2c120ffb4..a41c7f146 100644 --- a/src/expression.rs +++ b/src/expression.rs @@ -143,33 +143,61 @@ impl<'a> Tree<'a> { )), // Function call Found::LBracket(n) => { - let mut ret = Tree { - name: &sl[..n], - args: vec![], - }; - - sl = &sl[n + 1..]; - loop { - let (arg, new_sl) = Tree::from_slice_delim(sl, depth + 1, delim)?; - ret.args.push(arg); - - if new_sl.is_empty() { + // Special case for musig + // We want the complex musig to be just one node in our tree. + if &sl[..n] == "musig" { + let mut open_brackets = 0; + let mut finish_index = 0; + for (i, ch) in sl[n..].char_indices() { + if ch == delim { + open_brackets = open_brackets + 1; + } else if ch == closing_delim(delim) { + open_brackets = open_brackets - 1; + } + if open_brackets == 0 { + finish_index = i; + break; + } + } + if open_brackets != 0 { return Err(Error::ExpectedChar(closing_delim(delim))); } + Ok(( + Tree { + name: &sl[..n + finish_index + 1], + args: vec![], + }, + &sl[n + finish_index + 1..], + )) + } else { + let mut ret = Tree { + name: &sl[..n], + args: vec![], + }; + // update the string to parse + sl = &sl[n + 1..]; + loop { + let (arg, new_sl) = Tree::from_slice_delim(sl, depth + 1, delim)?; + ret.args.push(arg); + // There has to be a closing bracket for the open bracket we encounter in this case + if new_sl.is_empty() { + return Err(Error::ExpectedChar(closing_delim(delim))); + } - sl = &new_sl[1..]; - match new_sl.as_bytes()[0] { - b',' => {} - last_byte => { - if last_byte == closing_delim(delim) as u8 { - break; - } else { - return Err(Error::ExpectedChar(closing_delim(delim))); + sl = &new_sl[1..]; + match new_sl.as_bytes()[0] { + b',' => {} + last_byte => { + if last_byte == closing_delim(delim) as u8 { + break; + } else { + return Err(Error::ExpectedChar(closing_delim(delim))); + } } } } + Ok((ret, sl)) } - Ok((ret, sl)) } } } diff --git a/src/miniscript/astelem.rs b/src/miniscript/astelem.rs index 932d5135e..c3dfd3fa8 100644 --- a/src/miniscript/astelem.rs +++ b/src/miniscript/astelem.rs @@ -412,16 +412,20 @@ impl_from_tree!( ;Ctx; ScriptContext, Terminal, fn from_tree(top: &expression::Tree) -> Result, Error> { + dbg!("from_tree of Terminal { frag_name = ""; frag_wrap = ""; } (Some(name), None, _) => { + dbg!("inside pk and pkh case"); if name == "pk" { frag_name = "pk_k"; frag_wrap = "c"; @@ -452,6 +456,7 @@ impl_from_tree!( frag_wrap = wrap; } } + // Why is this error ?? consider s:c:pk() two times wrapped (Some(_), Some(_), Some(_)) => { return Err(Error::MultiColon(top.name.to_owned())); } @@ -551,9 +556,11 @@ impl_from_tree!( top.args.len(), ))), }?; + dbg!(&unwrapped); for ch in frag_wrap.chars().rev() { // Check whether the wrapper is valid under the current context let ms = Miniscript::from_ast(unwrapped)?; + dbg!(&ms); Ctx::check_global_validity(&ms)?; match ch { 'a' => unwrapped = Terminal::Alt(Arc::new(ms)), diff --git a/src/miniscript/mod.rs b/src/miniscript/mod.rs index 08c7f75d6..7be449412 100644 --- a/src/miniscript/mod.rs +++ b/src/miniscript/mod.rs @@ -139,6 +139,7 @@ impl fmt::Display for Miniscript impl Miniscript { /// Extracts the `AstElem` representing the root of the miniscript pub fn into_inner(self) -> Terminal { + dbg!("into inner of Miniscript Result, Error> { + dbg!("from_tree inside impl of Miniscript"); let inner: Terminal = expression::FromTree::from_tree(top)?; + dbg!(&inner); Ok(Miniscript { ty: Type::type_check(&inner, |_| None)?, ext: ExtData::type_check(&inner, |_| None)?,