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#[derive(Debug, PartialEq, Eq)]
19pub enum ParseError {
20 InvalidEncoding,
22 NotZcash,
24 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 fn from_str(s: &str) -> Result<Self, Self::Err> {
56 let s = s.trim();
58
59 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 }
70 Err(e) => {
71 return Err(ParseError::from(e));
72 }
73 }
74
75 if let Ok(parsed) = CheckedHrpstring::new::<Bech32>(s) {
77 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 _ => {
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 if let Ok(parsed) = CheckedHrpstring::new::<Bech32m>(s) {
99 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 _ => {
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 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 _ => 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 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 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 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 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 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}