Skip to main content

zcash_address/
encoding.rs

1use alloc::string::String;
2use alloc::vec::Vec;
3use core::convert::TryInto;
4use core::fmt;
5use core::str::FromStr;
6
7#[cfg(feature = "std")]
8use std::error::Error;
9
10use bech32::{Bech32, Bech32m, Checksum, Hrp, primitives::decode::CheckedHrpstring};
11use zcash_protocol::consensus::{NetworkConstants, NetworkType};
12use zcash_protocol::constants::{mainnet, regtest, testnet};
13
14use crate::kind::unified::Encoding;
15use crate::{AddressKind, ZcashAddress, kind::*};
16
17/// An error while attempting to parse a string as a Zcash address.
18#[derive(Debug, PartialEq, Eq)]
19pub enum ParseError {
20    /// The string is an invalid encoding.
21    InvalidEncoding,
22    /// The string is not a Zcash address.
23    NotZcash,
24    /// Errors specific to unified addresses.
25    Unified(unified::ParseError),
26}
27
28impl From<unified::ParseError> for ParseError {
29    fn from(e: unified::ParseError) -> Self {
30        match e {
31            unified::ParseError::InvalidEncoding(_) => Self::InvalidEncoding,
32            unified::ParseError::UnknownPrefix(_) => Self::NotZcash,
33            _ => Self::Unified(e),
34        }
35    }
36}
37
38impl fmt::Display for ParseError {
39    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
40        match self {
41            ParseError::InvalidEncoding => write!(f, "Invalid encoding"),
42            ParseError::NotZcash => write!(f, "Not a Zcash address"),
43            ParseError::Unified(e) => e.fmt(f),
44        }
45    }
46}
47
48#[cfg(feature = "std")]
49impl Error for ParseError {}
50
51impl FromStr for ZcashAddress {
52    type Err = ParseError;
53
54    /// Attempts to parse the given string as a Zcash address.
55    fn from_str(s: &str) -> Result<Self, Self::Err> {
56        // Remove leading and trailing whitespace, to handle copy-paste errors.
57        let s = s.trim();
58
59        // Try decoding as a unified address
60        match unified::Address::decode(s) {
61            Ok((net, data)) => {
62                return Ok(ZcashAddress {
63                    net,
64                    kind: AddressKind::Unified(data),
65                });
66            }
67            Err(unified::ParseError::NotUnified | unified::ParseError::UnknownPrefix(_)) => {
68                // allow decoding to fall through to Sapling/TEX/Transparent
69            }
70            Err(e) => {
71                return Err(ParseError::from(e));
72            }
73        }
74
75        // Try decoding as a Sapling address (Bech32)
76        if let Ok(parsed) = CheckedHrpstring::new::<Bech32>(s) {
77            // If we reached this point, the encoding is found to be valid Bech32.
78            let net = match parsed.hrp().as_str() {
79                mainnet::HRP_SAPLING_PAYMENT_ADDRESS => NetworkType::Main,
80                testnet::HRP_SAPLING_PAYMENT_ADDRESS => NetworkType::Test,
81                regtest::HRP_SAPLING_PAYMENT_ADDRESS => NetworkType::Regtest,
82                // We will not define new Bech32 address encodings.
83                _ => {
84                    return Err(ParseError::NotZcash);
85                }
86            };
87
88            let data = parsed.byte_iter().collect::<Vec<_>>();
89
90            return data
91                .try_into()
92                .map(AddressKind::Sapling)
93                .map_err(|_| ParseError::InvalidEncoding)
94                .map(|kind| ZcashAddress { net, kind });
95        }
96
97        // Try decoding as a TEX address (Bech32m)
98        if let Ok(parsed) = CheckedHrpstring::new::<Bech32m>(s) {
99            // If we reached this point, the encoding is found to be valid Bech32m.
100            let net = match parsed.hrp().as_str() {
101                mainnet::HRP_TEX_ADDRESS => NetworkType::Main,
102                testnet::HRP_TEX_ADDRESS => NetworkType::Test,
103                regtest::HRP_TEX_ADDRESS => NetworkType::Regtest,
104                // Not recognized as a Zcash address type
105                _ => {
106                    return Err(ParseError::NotZcash);
107                }
108            };
109
110            let data = parsed.byte_iter().collect::<Vec<_>>();
111
112            return data
113                .try_into()
114                .map(AddressKind::Tex)
115                .map_err(|_| ParseError::InvalidEncoding)
116                .map(|kind| ZcashAddress { net, kind });
117        }
118
119        // The rest use Base58Check.
120        if let Ok(decoded) = bs58::decode(s).with_check(None).into_vec() {
121            if decoded.len() >= 2 {
122                let (prefix, net) = match decoded[..2].try_into().unwrap() {
123                    prefix @ (mainnet::B58_PUBKEY_ADDRESS_PREFIX
124                    | mainnet::B58_SCRIPT_ADDRESS_PREFIX
125                    | mainnet::B58_SPROUT_ADDRESS_PREFIX) => (prefix, NetworkType::Main),
126                    prefix @ (testnet::B58_PUBKEY_ADDRESS_PREFIX
127                    | testnet::B58_SCRIPT_ADDRESS_PREFIX
128                    | testnet::B58_SPROUT_ADDRESS_PREFIX) => (prefix, NetworkType::Test),
129                    // We will not define new Base58Check address encodings.
130                    _ => return Err(ParseError::NotZcash),
131                };
132
133                return match prefix {
134                    mainnet::B58_SPROUT_ADDRESS_PREFIX | testnet::B58_SPROUT_ADDRESS_PREFIX => {
135                        decoded[2..].try_into().map(AddressKind::Sprout)
136                    }
137                    mainnet::B58_PUBKEY_ADDRESS_PREFIX | testnet::B58_PUBKEY_ADDRESS_PREFIX => {
138                        decoded[2..].try_into().map(AddressKind::P2pkh)
139                    }
140                    mainnet::B58_SCRIPT_ADDRESS_PREFIX | testnet::B58_SCRIPT_ADDRESS_PREFIX => {
141                        decoded[2..].try_into().map(AddressKind::P2sh)
142                    }
143                    _ => unreachable!(),
144                }
145                .map_err(|_| ParseError::InvalidEncoding)
146                .map(|kind| ZcashAddress { kind, net });
147            }
148        };
149
150        // If it's not valid Bech32, Bech32m, or Base58Check, it's not a Zcash address.
151        Err(ParseError::NotZcash)
152    }
153}
154
155fn encode_bech32<Ck: Checksum>(hrp: &str, data: &[u8]) -> String {
156    bech32::encode::<Ck>(Hrp::parse_unchecked(hrp), data).expect("encoding is short enough")
157}
158
159fn encode_b58(prefix: [u8; 2], data: &[u8]) -> String {
160    let mut bytes = Vec::with_capacity(2 + data.len());
161    bytes.extend_from_slice(&prefix);
162    bytes.extend_from_slice(data);
163    bs58::encode(bytes).with_check().into_string()
164}
165
166impl fmt::Display for ZcashAddress {
167    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
168        let encoded = match &self.kind {
169            AddressKind::Sprout(data) => encode_b58(self.net.b58_sprout_address_prefix(), data),
170            AddressKind::Sapling(data) => {
171                encode_bech32::<Bech32>(self.net.hrp_sapling_payment_address(), data)
172            }
173            AddressKind::Unified(addr) => addr.encode(&self.net),
174            AddressKind::P2pkh(data) => encode_b58(self.net.b58_pubkey_address_prefix(), data),
175            AddressKind::P2sh(data) => encode_b58(self.net.b58_script_address_prefix(), data),
176            AddressKind::Tex(data) => encode_bech32::<Bech32m>(self.net.hrp_tex_address(), data),
177        };
178        write!(f, "{}", encoded)
179    }
180}
181
182#[cfg(test)]
183mod tests {
184    use alloc::string::ToString;
185
186    use assert_matches::assert_matches;
187
188    use super::*;
189    use crate::kind::unified;
190    use zcash_protocol::consensus::NetworkType;
191
192    fn encoding(encoded: &str, decoded: ZcashAddress) {
193        assert_eq!(decoded.to_string(), encoded);
194        assert_eq!(encoded.parse(), Ok(decoded));
195    }
196
197    #[test]
198    fn sprout() {
199        encoding(
200            "zc8E5gYid86n4bo2Usdq1cpr7PpfoJGzttwBHEEgGhGkLUg7SPPVFNB2AkRFXZ7usfphup5426dt1buMmY3fkYeRrQGLa8y",
201            ZcashAddress {
202                net: NetworkType::Main,
203                kind: AddressKind::Sprout([0; 64]),
204            },
205        );
206        encoding(
207            "ztJ1EWLKcGwF2S4NA17pAJVdco8Sdkz4AQPxt1cLTEfNuyNswJJc2BbBqYrsRZsp31xbVZwhF7c7a2L9jsF3p3ZwRWpqqyS",
208            ZcashAddress {
209                net: NetworkType::Test,
210                kind: AddressKind::Sprout([0; 64]),
211            },
212        );
213    }
214
215    #[test]
216    fn sapling() {
217        encoding(
218            "zs1qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqpq6d8g",
219            ZcashAddress {
220                net: NetworkType::Main,
221                kind: AddressKind::Sapling([0; 43]),
222            },
223        );
224        encoding(
225            "ztestsapling1qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqfhgwqu",
226            ZcashAddress {
227                net: NetworkType::Test,
228                kind: AddressKind::Sapling([0; 43]),
229            },
230        );
231        encoding(
232            "zregtestsapling1qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqknpr3m",
233            ZcashAddress {
234                net: NetworkType::Regtest,
235                kind: AddressKind::Sapling([0; 43]),
236            },
237        );
238    }
239
240    #[test]
241    fn unified() {
242        encoding(
243            "u1qpatys4zruk99pg59gcscrt7y6akvl9vrhcfyhm9yxvxz7h87q6n8cgrzzpe9zru68uq39uhmlpp5uefxu0su5uqyqfe5zp3tycn0ecl",
244            ZcashAddress {
245                net: NetworkType::Main,
246                kind: AddressKind::Unified(unified::Address(vec![
247                    unified::address::Receiver::Sapling([0; 43]),
248                ])),
249            },
250        );
251        encoding(
252            "utest10c5kutapazdnf8ztl3pu43nkfsjx89fy3uuff8tsmxm6s86j37pe7uz94z5jhkl49pqe8yz75rlsaygexk6jpaxwx0esjr8wm5ut7d5s",
253            ZcashAddress {
254                net: NetworkType::Test,
255                kind: AddressKind::Unified(unified::Address(vec![
256                    unified::address::Receiver::Sapling([0; 43]),
257                ])),
258            },
259        );
260        encoding(
261            "uregtest15xk7vj4grjkay6mnfl93dhsflc2yeunhxwdh38rul0rq3dfhzzxgm5szjuvtqdha4t4p2q02ks0jgzrhjkrav70z9xlvq0plpcjkd5z3",
262            ZcashAddress {
263                net: NetworkType::Regtest,
264                kind: AddressKind::Unified(unified::Address(vec![
265                    unified::address::Receiver::Sapling([0; 43]),
266                ])),
267            },
268        );
269
270        let badencoded = "uinvalid1ck5navqwcng43gvsxwrxsplc22p7uzlcag6qfa0zh09e87efq6rq8wsnv25umqjjravw70rl994n5ueuhza2fghge5gl7zrl2qp6cwmp";
271        assert_eq!(
272            badencoded.parse::<ZcashAddress>(),
273            Err(ParseError::NotZcash)
274        );
275    }
276
277    #[test]
278    fn transparent() {
279        encoding(
280            "t1Hsc1LR8yKnbbe3twRp88p6vFfC5t7DLbs",
281            ZcashAddress {
282                net: NetworkType::Main,
283                kind: AddressKind::P2pkh([0; 20]),
284            },
285        );
286        encoding(
287            "tm9iMLAuYMzJ6jtFLcA7rzUmfreGuKvr7Ma",
288            ZcashAddress {
289                net: NetworkType::Test,
290                kind: AddressKind::P2pkh([0; 20]),
291            },
292        );
293        encoding(
294            "t3JZcvsuaXE6ygokL4XUiZSTrQBUoPYFnXJ",
295            ZcashAddress {
296                net: NetworkType::Main,
297                kind: AddressKind::P2sh([0; 20]),
298            },
299        );
300        encoding(
301            "t26YoyZ1iPgiMEWL4zGUm74eVWfhyDMXzY2",
302            ZcashAddress {
303                net: NetworkType::Test,
304                kind: AddressKind::P2sh([0; 20]),
305            },
306        );
307    }
308
309    #[test]
310    fn tex() {
311        let p2pkh_str = "t1VmmGiyjVNeCjxDZzg7vZmd99WyzVby9yC";
312        let tex_str = "tex1s2rt77ggv6q989lr49rkgzmh5slsksa9khdgte";
313
314        // Transcode P2PKH to TEX
315        let p2pkh_zaddr: ZcashAddress = p2pkh_str.parse().unwrap();
316        assert_matches!(p2pkh_zaddr.net, NetworkType::Main);
317        if let AddressKind::P2pkh(zaddr_data) = p2pkh_zaddr.kind {
318            let tex_zaddr = ZcashAddress {
319                net: p2pkh_zaddr.net,
320                kind: AddressKind::Tex(zaddr_data),
321            };
322
323            assert_eq!(tex_zaddr.to_string(), tex_str);
324        } else {
325            panic!("Decoded address should have been a P2PKH address.");
326        }
327
328        // Transcode TEX to P2PKH
329        let tex_zaddr: ZcashAddress = tex_str.parse().unwrap();
330        assert_matches!(tex_zaddr.net, NetworkType::Main);
331        if let AddressKind::Tex(zaddr_data) = tex_zaddr.kind {
332            let p2pkh_zaddr = ZcashAddress {
333                net: tex_zaddr.net,
334                kind: AddressKind::P2pkh(zaddr_data),
335            };
336
337            assert_eq!(p2pkh_zaddr.to_string(), p2pkh_str);
338        } else {
339            panic!("Decoded address should have been a TEX address.");
340        }
341    }
342
343    #[test]
344    fn tex_testnet() {
345        let p2pkh_str = "tm9ofD7kHR7AF8MsJomEzLqGcrLCBkD9gDj";
346        let tex_str = "textest1qyqszqgpqyqszqgpqyqszqgpqyqszqgpfcjgfy";
347
348        // Transcode P2PKH to TEX
349        let p2pkh_zaddr: ZcashAddress = p2pkh_str.parse().unwrap();
350        assert_matches!(p2pkh_zaddr.net, NetworkType::Test);
351        if let AddressKind::P2pkh(zaddr_data) = p2pkh_zaddr.kind {
352            let tex_zaddr = ZcashAddress {
353                net: p2pkh_zaddr.net,
354                kind: AddressKind::Tex(zaddr_data),
355            };
356
357            assert_eq!(tex_zaddr.to_string(), tex_str);
358        } else {
359            panic!("Decoded address should have been a P2PKH address.");
360        }
361
362        // Transcode TEX to P2PKH
363        let tex_zaddr: ZcashAddress = tex_str.parse().unwrap();
364        assert_matches!(tex_zaddr.net, NetworkType::Test);
365        if let AddressKind::Tex(zaddr_data) = tex_zaddr.kind {
366            let p2pkh_zaddr = ZcashAddress {
367                net: tex_zaddr.net,
368                kind: AddressKind::P2pkh(zaddr_data),
369            };
370
371            assert_eq!(p2pkh_zaddr.to_string(), p2pkh_str);
372        } else {
373            panic!("Decoded address should have been a TEX address.");
374        }
375    }
376
377    #[test]
378    fn whitespace() {
379        assert_eq!(
380            " t1Hsc1LR8yKnbbe3twRp88p6vFfC5t7DLbs".parse(),
381            Ok(ZcashAddress {
382                net: NetworkType::Main,
383                kind: AddressKind::P2pkh([0; 20])
384            }),
385        );
386        assert_eq!(
387            "t1Hsc1LR8yKnbbe3twRp88p6vFfC5t7DLbs ".parse(),
388            Ok(ZcashAddress {
389                net: NetworkType::Main,
390                kind: AddressKind::P2pkh([0; 20])
391            }),
392        );
393        assert_eq!(
394            "something t1Hsc1LR8yKnbbe3twRp88p6vFfC5t7DLbs".parse::<ZcashAddress>(),
395            Err(ParseError::NotZcash),
396        );
397    }
398}