This document outlines a test suite specification that can be used to verify the correctness of a Sparse Merkle Tree's outputs. The scope of this document covers only Sparse Merkle Tree (SMT) implementations that are compliant with Celestia Sparse Merkle Tree Specification Icon Link. The goal of this document is to equip SMT library developers with a supplemental indicator of correctness. Libraries implementing an SMT can additionally implement this test suite specification in the code base's native language. Passing all tests in the concrete test suite is an indication of correctness and consistency with the reference specification; however, it is not an absolute guarantee.
The tests described in this document are designed to test features common to most Sparse Merkle Tree implementations. Test specifications are agnostic of the implementation details or language, and therefore take a black-box testing approach. A test specification may provide an example of what a compliant test may look like in the form of pseudocode.
A test specification follows the format:
Test name
Test description
Test inputs
Test outputs
Example pseudocode
For a concrete test to comply with its corresponding test specification, the System Under Test (SUT) must take in the prescribed inputs. When the SUT produces the prescribed outputs, the test passes. When the SUT produces any result or error that is not prescribed by the specification, the test fails. For a library to comply with the complete specification described herein, it must implement all test specifications, and each test must pass.
All test specifications assume that the Merkle Tree implementation under test uses the SHA-2-256 hashing algorithm as defined in FIPS PUB 180-4 Icon Link to produce its outputs. The following test cases stipulate a theoretical function Sum(N) that takes in a big endian data slice N and returns the 32 byte SHA-256 hash of N.
Tests the root after performing two update calls with the same inputs. The resulting input set is described by S = {A} U {A} = {A}, where {A} is the input. This test expects a root signature identical to that produced by Test Update 1 .
Inputs:
Update the empty tree with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"DATA" (bytes, UTF-8)
Update the tree again with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"DATA" (bytes, UTF-8)
Outputs:
The expected root signature: 0x39f36a7cb4dfb1b46f03d044265df6a491dffc1034121bc1071a34ddce9bb14b
Tests the root after performing two update calls with the same leaf keys but different leaf data. The second update call is expected to overwrite the data originally written by the first update call.
Inputs:
Update the empty tree with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"DATA" (bytes, UTF-8)
Update the tree with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"CHANGE" (bytes, UTF-8)
Outputs:
The expected root signature: 0xdd97174c80e5e5aa3a31c61b05e279c1495c8a07b2a08bca5dbc9fb9774f9457
Tests the root after performing update calls with discontinuous sets of inputs. The resulting input set is described by S = [0..5) U [10..15) U [20..25).
Inputs:
For each i in 0..5, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 10..15, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 20..25, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
Outputs:
The expected root signature: 0x7e6643325042cfe0fc76626c043b97062af51c7e9fc56665f12b479034bce326
Example Pseudocode:
smt = SparseMerkleTree.new(Storage.new(), sha256.new())for i in 0..5 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 10..15 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 20..25 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}root = smt.root()expected_root = '7e6643325042cfe0fc76626c043b97062af51c7e9fc56665f12b479034bce326'expect(hex_encode(root), expected_root).to_be_equal
Tests the root after performing one update call with empty data. Updating the empty tree with empty data does not change the root, and the expected root remains the default root. The resulting input set is described by S = {Ø} U {Ø} = {Ø}. This test expects a root signature identical to that produced by Test Empty Root .
Inputs:
Update the empty tree with (K, D), where leaf key K = Sum(0u32) and empty leaf data D = b"" (0 bytes)
Outputs:
The expected root signature: 0x0000000000000000000000000000000000000000000000000000000000000000
Tests the root after performing one update call with arbitrary data followed by a second update call on the same key with empty data. Updating a key with empty data is equivalent to calling delete. By deleting the only key, we have an empty tree and expect to arrive at the default root. The resulting input set is described by S = {0} - {0} = {Ø}. This test expects a root signature identical to that produced by Test Empty Root .
Inputs:
Update the empty tree with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"DATA" (bytes, UTF-8)
Update the tree with (K, D), where leaf key K = Sum(0u32) and empty leaf data D = b"" (0 bytes)
Outputs:
The expected root signature: 0x0000000000000000000000000000000000000000000000000000000000000000
Tests the root after performing one update call followed by a subsequent delete call on the same key. By deleting the only key, we have an empty tree and expect to arrive at the default root. The resulting input set is described by S = {0} - {0} = {Ø}. This test expects a root signature identical to that produced by Test Empty Root .
Inputs:
Update the empty tree with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"DATA" (bytes, UTF-8)
Delete (K) from the tree, where leaf key K = Sum(0u32)
Outputs:
The expected root signature: 0x0000000000000000000000000000000000000000000000000000000000000000
Tests the root after performing two update calls followed by a subsequent delete call on the first key. By deleting the second key, we have a tree with only one key remaining, equivalent to a single update. This test expects a root signature identical to that produced by Test Update 1 .
Inputs:
Update the empty tree with (K, D), where leaf key K = Sum(0u32) and leaf data D = b"DATA" (bytes, UTF-8)
Update the tree with (K, D), where leaf key K = Sum(1u32) and leaf data D = b"DATA" (bytes, UTF-8)
Delete (K) from the tree, where leaf key K = Sum(1u32)
Outputs:
The expected root signature: 0x39f36a7cb4dfb1b46f03d044265df6a491dffc1034121bc1071a34ddce9bb14b
Tests the root after performing 10 update calls followed by 5 subsequent delete calls on the latter keys. By deleting the last five keys, we have a tree with the first five keys remaining, equivalent to five updates. This test expects a root signature identical to that produced by Test Update 5 .
Inputs:
For each i in 0..10, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 5..10, delete (K) from the tree, where leaf key K = Sum(i)
Outputs:
The expected root signature: 0x108f731f2414e33ae57e584dc26bd276db07874436b2264ca6e520c658185c6b
Example Pseudocode:
smt = SparseMerkleTree.new(Storage.new(), sha256.new())for i in 0..10 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 5..10 { key = &(i as u32).to_big_endian_bytes() smt.delete(&sum(key))}root = smt.root()expected_root = '108f731f2414e33ae57e584dc26bd276db07874436b2264ca6e520c658185c6b'expect(hex_encode(root), expected_root).to_be_equal
Tests the root after performing five update calls followed by a subsequent delete on a key that is not present in the input set. This test expects a root signature identical to that produced by Test Update 5 .
Inputs:
For each i in 0..5, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
Delete (K) from the tree, where leaf key K = Sum(1024u32)
Outputs:
The expected root signature: 0x108f731f2414e33ae57e584dc26bd276db07874436b2264ca6e520c658185c6b
Example Pseudocode:
smt = SparseMerkleTree.new(Storage.new(), sha256.new())for i in 0..5 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}smt.delete(&sum(b"\x00\x00\x04\x00"))root = smt.root()expected_root = '108f731f2414e33ae57e584dc26bd276db07874436b2264ca6e520c658185c6b'expect(hex_encode(root), expected_root).to_be_equal
Tests the root after performing a series of interleaved update and delete calls. The resulting input set is described by [0..5) U [10..15) U [20..25). This test demonstrates the inverse relationship between operations update and delete. This test expects a root signature identical to that produced by Test Update Union .
Inputs:
For each i in 0..10, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 5..15, delete (K) from the tree, where leaf key K = Sum(i) from the tree
For each i in 10..20, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 15..25, delete (K) from the tree, where leaf key K = Sum(i) from the tree
For each i in 20..30, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 25..35, delete (K) from the tree, where leaf key K = Sum(i) from the tree
Outputs:
The expected root signature: 0x7e6643325042cfe0fc76626c043b97062af51c7e9fc56665f12b479034bce326
Example Pseudocode:
smt = SparseMerkleTree.new(Storage.new(), sha256.new())for i in 0..10 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 5..15 { key = &(i as u32).to_big_endian_bytes() smt.delete(&sum(key))}for i in 10..20 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 15..25 { key = &(i as u32).to_big_endian_bytes() smt.delete(&sum(key))}for i in 20..30 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 25..35 { key = &(i as u32).to_big_endian_bytes() smt.delete(&sum(key))}root = smt.root()expected_root = '7e6643325042cfe0fc76626c043b97062af51c7e9fc56665f12b479034bce326'expect(hex_encode(root), expected_root).to_be_equal
Tests the root after performing delete calls with discontinuous sets of inputs. The resulting input set is described by S = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] - [1, 3, 5, 7, 9] = [0, 2, 4, 6, 8]. This test expects a root signature identical to that produced by Test Update Sparse Union .
Inputs:
For each i in 0..10, update the tree with (K, D), where leaf key K = Sum(i) and leaf data D = b"DATA" (bytes, UTF-8)
For each i in 0..5, delete (K) from the tree, where leaf key K = Sum(i * 2 + 1)
Outputs:
The expected root signature: 0xe912e97abc67707b2e6027338292943b53d01a7fbd7b244674128c7e468dd696
Example Pseudocode:
smt = SparseMerkleTree.new(Storage.new(), sha256.new())for i in 0..10 { key = &(i as u32).to_big_endian_bytes() data = b"DATA" smt.update(&sum(key), data)}for i in 0..5 { key = &(i as u32 * 2 + 1).to_big_endian_bytes() smt.delete(&sum(key))}root = smt.root()expected_root = 'e912e97abc67707b2e6027338292943b53d01a7fbd7b244674128c7e468dd696'expect(hex_encode(root), expected_root).to_be_equal