snowbridge_ethereum/
mpt.rs1use ethereum_types::H256;
6use sp_std::prelude::*;
7
8pub trait Node {
9 fn contains_hash(&self, hash: H256) -> bool;
10}
11
12impl TryFrom<&[u8]> for Box<dyn Node> {
13 type Error = rlp::DecoderError;
14
15 fn try_from(bytes: &[u8]) -> Result<Box<dyn Node>, Self::Error> {
16 let rlp = rlp::Rlp::new(bytes);
17 match rlp.item_count()? {
18 2 => {
19 let node: ShortNode = rlp.as_val()?;
20 Ok(Box::new(node))
21 },
22 17 => {
23 let node: FullNode = rlp.as_val()?;
24 Ok(Box::new(node))
25 },
26 _ => Err(rlp::DecoderError::Custom("Invalid number of list elements")),
27 }
28 }
29}
30
31pub struct FullNode {
35 pub children: Vec<Option<H256>>,
36}
37
38impl rlp::Decodable for FullNode {
39 fn decode(rlp: &rlp::Rlp) -> Result<Self, rlp::DecoderError> {
40 let children: Vec<Option<H256>> = rlp
41 .iter()
42 .map(|item| {
43 let v: Vec<u8> = item.as_val()?;
44 match v.len() {
45 0 => Ok(None),
46 32 => {
47 let mut bytes = [0u8; 32];
48 bytes.copy_from_slice(&v);
49 Ok(Some(bytes.into()))
50 },
51 _ => Err(rlp::DecoderError::Custom("Expected 32-byte hash or empty child")),
52 }
53 })
54 .collect::<Result<_, rlp::DecoderError>>()?;
55
56 Ok(Self { children })
57 }
58}
59
60impl Node for FullNode {
61 fn contains_hash(&self, hash: H256) -> bool {
62 self.children.iter().any(|h| Some(hash) == *h)
63 }
64}
65
66pub struct ShortNode {
71 pub key: Vec<u8>,
72 pub value: Vec<u8>,
73}
74
75impl rlp::Decodable for ShortNode {
76 fn decode(rlp: &rlp::Rlp) -> Result<Self, rlp::DecoderError> {
77 let mut iter = rlp.iter();
78
79 let key: Vec<u8> = match iter.next() {
80 Some(data) => data.as_val()?,
81 None => return Err(rlp::DecoderError::Custom("Expected key bytes")),
82 };
83
84 let value: Vec<u8> = match iter.next() {
85 Some(data) => data.as_val()?,
86 None => return Err(rlp::DecoderError::Custom("Expected value bytes")),
87 };
88
89 Ok(Self { key, value })
90 }
91}
92
93impl Node for ShortNode {
94 fn contains_hash(&self, hash: H256) -> bool {
95 self.value == hash.0
96 }
97}
98
99#[cfg(test)]
100mod tests {
101
102 use super::*;
103 use hex_literal::hex;
104
105 const RAW_PROOF: [&[u8]; 3] = [
106 &hex!("f90131a0b5ba404eb5a6a88e56579f4d37ef9813b5ad7f86f0823ff3b407ac5a6bb465eca0398ead2655e78e03c127ce22c5830e90f18b1601ec055f938336c084feb915a9a026d322c26e46c50942c1aabde50e36df5cde572aed650ce73ea3182c6e90a02ca00600a356135f4db1db0d9842264cdff2652676f881669e91e316c0b6dd783011a0837f1deb4075336da320388c1edfffc56c448a43f4a5ba031300d32a7b509fc5a01c3ac82fd65b4aba7f9afaf604d9c82ec7e2deb573a091ae235751bc5c0c288da05d454159d9071b0f68b6e0503d290f23ac7602c1db0c569dee4605d8f5298f09a00bbed10350ec954448df795f6fd46e3faefc800ede061b3840eedc6e2b07a74da0acb02d26a3650f2064c14a435fdf1f668d8655daf455ebdf671713a7c089b3898080808080808080"),
107 &hex!("f901f180a00046a08d4f0bdbdc6b31903086ce323182bce6725e7d9415f7ff91ee8f4820bda0e7cd26ad5f3d2771e4b5ab788e268a14a10209f94ee918eb6c829d21d3d11c1da00d4a56d9e9a6751874fd86c7e3cb1c6ad5a848da62751325f478978a00ea966ea064b81920c8f04a8a1e21f53a8280e739fbb7b00b2ab92493ca3f610b70e8ac85a0b1040ed4c55a73178b76abb16f946ce5bebd6b93ab873c83327df54047d12c27a0de6485e9ac58dc6e2b04b4bb38f562684f0b1a2ee586cc11079e7d9a9dc40b32a0d394f4d3532c3124a65fa36e69147e04fd20453a72ee9c50660f17e13ce9df48a066501003fc3e3478efd2803cd0eded6bbe9243ca01ba754d6327071ddbcbc649a0b2684e518f325fee39fc8ea81b68f3f5c785be00d087f3bed8857ae2ee8da26ea071060a5c52042e8d7ce21092f8ecf06053beb9a0b773a6f91a30c4220aa276b2a0fc22436632574ccf6043d0986dede27ea94c9ca9a3bb5ec03ce776a4ddef24a9a05a8a1d6698c4e7d8cc3a2506cb9b12ea9a079c9c7099bc919dc804033cc556e4a0170c468b0716fd36d161f0bf05875f15756a2976de92f9efe7716320509d79c9a0182f909a90cab169f3efb62387f9cccdd61440acc4deec42f68a4f7ca58075c7a055cf0e9202ac75689b76318f1171f3a44465eddc06aae0713bfb6b34fdd27b7980"),
108 &hex!("f904de20b904daf904d701830652f0b9010004200000000000000000000080020000000000010000000000010000000000000000000000000000000000000000000002000000080000000000000000200000000000000000000000000008000000220000000000400010000000000000000000000000000000000000000000000000000000000000040000000010000100000000000800000000004000000000000000000000000000080000004000000000020000000000020000000000000000000000000000000000000000000004000000000002000000000100000000000000000000000000001000000002000020000010200000000000010000000000000000000000000000000000000010000000f903ccf89b9421130f34829b4c343142047a28ce96ec07814b15f863a0ddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa00000000000000000000000007d843005c7433c16b27ff939cb37471541561ebda0000000000000000000000000e9c1281aae66801fa35ec404d5f2aea393ff6988a000000000000000000000000000000000000000000000000000000005d09b7380f89b9421130f34829b4c343142047a28ce96ec07814b15f863a08c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925a00000000000000000000000007d843005c7433c16b27ff939cb37471541561ebda00000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488da0ffffffffffffffffffffffffffffffffffffffffffffffffffffffcc840c6920f89b94c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2f863a0ddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa0000000000000000000000000e9c1281aae66801fa35ec404d5f2aea393ff6988a00000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488da000000000000000000000000000000000000000000000000003e973b5a5d1078ef87994e9c1281aae66801fa35ec404d5f2aea393ff6988e1a01c411e9a96e071241c2f21f7726b17ae89e3cab4c78be50e062b03a9fffbbad1b840000000000000000000000000000000000000000000000000000001f1420ad1d40000000000000000000000000000000000000000000000014ad400879d159a38f8fc94e9c1281aae66801fa35ec404d5f2aea393ff6988f863a0d78ad95fa46c994b6551d0da85fc275fe613ce37657fb8d5e3d130840159d822a00000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488da00000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488db88000000000000000000000000000000000000000000000000000000005d415f3320000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000003e973b5a5d1078ef87a94c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2f842a07fcf532c15f0a6db0bd6d0e038bea71d30d808c7d98cb3bf7268a95bf5081b65a00000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488da000000000000000000000000000000000000000000000000003e973b5a5d1078e"),
109 ];
110
111 #[test]
112 fn decode_full_node() {
113 let node1: FullNode = rlp::decode(RAW_PROOF[0]).unwrap();
114 let node2: FullNode = rlp::decode(RAW_PROOF[1]).unwrap();
115 assert_eq!(node1.children.len(), 17);
116 assert_eq!(node2.children.len(), 17);
117 assert_eq!(node1.children.iter().filter(|c| c.is_none()).count(), 8);
118 assert_eq!(node2.children.iter().filter(|c| c.is_none()).count(), 2);
119
120 let result: Result<FullNode, rlp::DecoderError> = rlp::decode(RAW_PROOF[2]);
121 assert!(result.is_err());
122 }
123
124 #[test]
125 fn decode_short_node() {
126 let node: ShortNode = rlp::decode(RAW_PROOF[2]).unwrap();
128 assert_eq!(node.key, vec![32]);
129 assert!(!node.value.is_empty());
130
131 let node: ShortNode = rlp::decode(&hex!(
133 "e4820001a04fff54398cad4d05ea6abfd8b0f3b4fe14c04d7ff5f5211c5b927d9cf72ac1d8"
134 ))
135 .unwrap();
136 assert_eq!(node.key, vec![0, 1]);
137 assert_eq!(
138 node.value,
139 hex!("4fff54398cad4d05ea6abfd8b0f3b4fe14c04d7ff5f5211c5b927d9cf72ac1d8").to_vec()
140 );
141 }
142}