coldcard 0.13.0

Coldcard Wallet Interface Library in Rust
Documentation
//! Derivation path module.
use std::fmt::Write;

/// BIP32 derivation path.
#[derive(Debug, Default)]
pub struct DerivationPath(Box<[Child]>);

impl DerivationPath {
    pub fn new(value: &str) -> Result<Self, Error> {
        let mut segments = value.split('/');
        match segments.next() {
            Some("m") => (),
            _ => return Err(Error::InvalidFormat),
        }

        let children: Box<[Child]> = segments.map(|c| c.parse()).collect::<Result<_, _>>()?;

        if children.len() > 12 {
            return Err(Error::TooDeep);
        }

        Ok(Self(children))
    }

    pub fn children(&self) -> &[Child] {
        &self.0
    }
}

impl std::fmt::Display for DerivationPath {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        write!(f, "m")?;
        for child in self.children() {
            write!(f, "/{}", child)?;
        }
        Ok(())
    }
}

impl TryFrom<Vec<Child>> for DerivationPath {
    type Error = Error;

    fn try_from(value: Vec<Child>) -> Result<Self, Self::Error> {
        if value.len() > 12 {
            Err(Error::TooDeep)
        } else {
            Ok(DerivationPath(value.into()))
        }
    }
}

/// Derivation path child segment.
#[derive(Debug, PartialEq, Eq)]
pub enum Child {
    Normal(u32),
    Hardened(u32),
}

impl Child {
    pub fn value(&self) -> u32 {
        match self {
            Child::Normal(i) => *i,
            Child::Hardened(i) => i | (1 << 31),
        }
    }
}

impl std::str::FromStr for Child {
    type Err = Error;

    fn from_str(c: &str) -> Result<Self, Self::Err> {
        let is_hardened = c.chars().last().is_some_and(|l| l == '\'' || l == 'h');
        let i: u32 = (if is_hardened { &c[0..c.len() - 1] } else { c })
            .parse()
            .map_err(|_| Error::InvalidChild)?;

        if i & (1 << 31) == 0 {
            if is_hardened {
                Ok(Child::Hardened(i))
            } else {
                Ok(Child::Normal(i))
            }
        } else {
            Err(Error::InvalidChild)
        }
    }
}

impl std::fmt::Display for Child {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Child::Normal(i) => std::fmt::Display::fmt(&i, f),
            Child::Hardened(i) => {
                std::fmt::Display::fmt(&i, f)?;
                f.write_char('\'')
            }
        }
    }
}

/// Derivation path error.
#[derive(Debug)]
pub enum Error {
    InvalidChild,
    InvalidFormat,
    TooDeep,
}

#[cfg(test)]
mod test {
    use super::*;

    #[test]
    fn parse() {
        let path = DerivationPath::new("m/44'/1'/2'/3/4").unwrap();
        let mut path = path.0.iter();
        assert_eq!(Some(&Child::Hardened(44)), path.next());
        assert_eq!(Some(&Child::Hardened(1)), path.next());
        assert_eq!(Some(&Child::Hardened(2)), path.next());
        assert_eq!(Some(&Child::Normal(3)), path.next());
        assert_eq!(Some(&Child::Normal(4)), path.next());
        assert_eq!(None, path.next());
    }

    #[test]
    fn display() {
        let path = DerivationPath::new("m/44'/1'/2'/3/4").unwrap();
        assert_eq!(path.to_string(), "m/44'/1'/2'/3/4");
    }
}