Skip to content

Commit 62f896d

Browse files
committed
Add initial support for KeyExpr
1 parent d589fe9 commit 62f896d

File tree

2 files changed

+102
-0
lines changed

2 files changed

+102
-0
lines changed

src/miniscript/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ pub mod hash256;
4141
pub mod iter;
4242
pub mod lex;
4343
pub mod limits;
44+
pub mod musig_key;
4445
pub mod satisfy;
4546
pub mod types;
4647

src/miniscript/musig_key.rs

+101
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,101 @@
1+
//! Support for multi-signature keys
2+
use std::str::FromStr;
3+
use std::{error, fmt};
4+
5+
use crate::expression::Tree;
6+
use crate::MiniscriptKey;
7+
8+
#[derive(Debug, Clone, PartialEq, PartialOrd, Ord, Eq, Hash)]
9+
/// Enum for representing keys in miniscript
10+
pub enum KeyExpr<Pk: MiniscriptKey> {
11+
/// Single-key (e.g pk(a), here 'a' is a single key)
12+
SingleKey(Pk),
13+
14+
/// Collection of keys in used for multi-signature
15+
MuSig(Vec<KeyExpr<Pk>>),
16+
}
17+
18+
#[derive(Debug, PartialEq)]
19+
/// Errors related to KeyExpr
20+
pub enum KeyExprError {
21+
/// Single key is empty string
22+
SingleKeyEmptyError,
23+
24+
/// Parsing error for single key
25+
SingleKeyParseError,
26+
}
27+
28+
impl fmt::Display for KeyExprError {
29+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
30+
match *self {
31+
KeyExprError::SingleKeyEmptyError => f.write_str("str for single key was empty"),
32+
KeyExprError::SingleKeyParseError => f.write_str("not able to parse the single key"),
33+
}
34+
}
35+
}
36+
37+
impl error::Error for KeyExprError {}
38+
39+
impl<Pk: MiniscriptKey + FromStr> FromStr for KeyExpr<Pk> {
40+
type Err = KeyExprError;
41+
fn from_str(s: &str) -> Result<Self, Self::Err> {
42+
let (key_tree, _) = Tree::from_slice(s).unwrap();
43+
fn expr_from_tree<Pk: MiniscriptKey + FromStr>(
44+
tree: &Tree,
45+
) -> Result<KeyExpr<Pk>, KeyExprError> {
46+
if tree.name == "musig" {
47+
let mut key_expr_vect = vec![];
48+
for sub_tree in tree.args.iter() {
49+
let temp_res = expr_from_tree(sub_tree)?;
50+
key_expr_vect.push(temp_res);
51+
}
52+
Ok(KeyExpr::MuSig(key_expr_vect))
53+
} else {
54+
if tree.name != "" {
55+
let single_key = match Pk::from_str(tree.name) {
56+
Ok(x) => x,
57+
Err(_) => {
58+
return Err(KeyExprError::SingleKeyParseError);
59+
}
60+
};
61+
Ok(KeyExpr::SingleKey(single_key))
62+
} else {
63+
Err(KeyExprError::SingleKeyEmptyError)
64+
}
65+
}
66+
}
67+
expr_from_tree(&key_tree)
68+
}
69+
}
70+
71+
impl<Pk: MiniscriptKey> fmt::Display for KeyExpr<Pk> {
72+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
73+
match *self {
74+
KeyExpr::SingleKey(ref pk) => write!(f, "{}", pk),
75+
KeyExpr::MuSig(ref my_vec) => {
76+
write!(f, "musig(")?;
77+
let len = my_vec.len();
78+
for (index, k) in my_vec.iter().enumerate() {
79+
if index == len - 1 {
80+
write!(f, "{}", k)?;
81+
} else {
82+
write!(f, "{},", k)?;
83+
}
84+
}
85+
f.write_str(")")
86+
}
87+
}
88+
}
89+
}
90+
#[cfg(test)]
91+
mod tests {
92+
use super::*;
93+
94+
#[test]
95+
fn test_one() {
96+
let dummy_key = "musig(A,B,musig(C,musig(D,E)))";
97+
let pk = KeyExpr::<String>::from_str(dummy_key).unwrap();
98+
println!("{}", pk);
99+
assert_eq!(dummy_key, format!("{}", pk))
100+
}
101+
}

0 commit comments

Comments
 (0)