Co-authored-by: markspanbroek <mark@spanbroek.net>
10 KiB
Reducing Metadata Overhead
Metadata plays a crucial role in any distributed or peer-to-peer (p2p) storage network. However, it often incurs significant overhead for the system. Therefore, it is important to understand the required metadata and how it should be stored, located, and transmitted.
Metadata and Manifests
Codex utilizes a metadata descriptor structure called the "manifest". A manifest is similar to a torrent file and stores various pieces of information necessary to describe a dataset.
Manifest
rootHash # Cid of root (tree) hash of the contained data set
originalBytes # Exact size of the original (uploaded) file
blockSize # Size of each contained block
blocks # Array of dataset blocks Cids
version # Cid version
hcodec # Multihash codec
codec # Data set codec
Additional information that describes erasure coding parameters may also be included:
Manifest
...
ecK # Number of blocks to encode
ecM # Number of resulting parity blocks
originalCid # The original Cid of the dataset being erasure coded
originalLen # The length of the original manifest
Manifests are treated as regular blocks of data, requiring no special handling by the Codex network or nodes. This means that announcing and storing manifests follows the same flow and uses the same subsystems as regular blocks. This convenience simplifies the execution flow significantly.
Manifest limitations
Including block hashes in the manifest introduces significant limitations. Firstly, the size of the manifest grows linearly with the number of hashes and the size of the hashing function itself, resulting in increased overhead for storing and transmitting manifests.
Overall, large manifests impose additional burden on the network in terms of storage and transmission, resulting in unnecessary overhead. For example, when retrieving a sizable file, it becomes necessary to obtain all the hashes listed in the manifest before downloading the initial block. This process can require hundreds of megabytes of data.
One way to reduce the number of hashes is to increase the block size, which only partially addresses the problem. A better solution however, is to completely remove the blocks array from the manifest and instead rely on a Merkle proofs to verify the block.
Slots and verification subsystem support
Besides the block hashes overhead, another reason for the change is the introduction of slots (verifiable dataset subsets) that nodes in a storage set/group store and verify. Slots require Merkle trees for verification, but otherwise are identical to the top-level dataset. Thus, storing and transmitting Merkle proofs is already a requirement for slot verification.
Replacing the blocks array with a proper Merkle tree would allow using the same mechanism proposed in this document, for both the top level dataset and for slot verification, storage and transmission. This greatly simplifies integration of the verification subsystem.
Removing blocks array
As already mentioned, the new mechanism proposed here, removes the blocks array from the manifest file in favor of a separate Merkle tree. This Merkle tree is persisted in the local store, and transmitted along side the dataset blocks on retrieval. This allows verifying the transmitted blocks without knowing it's hashes a priory.
Implementation overview
This mechanism requires an efficient Merkle tree implementation, which also allows persisting the leafs and intermediary hashes to disk; changes to the block exchange engine to support querying blocks by root hash and block index; and integration with the block store abstraction.
Merkle Tree
The block hashes array is replaced by a Merkle tree. The Merkle tree should support persisting to disk, partial and non blocking reads/writes, loading and storing from (async) iterators. For reference, checkout out https://github.com/filecoin-project/merkletree.
Block retrieval
Block Exchange Engine
The block exchange engine requires support for querying blocks by their index and respective dataset Merkle root. It also requires returning the Merkle proofs along side the chunk so that it can be readily verified. Scheduling blocks for retrieval should largely remain the same, but additional request and response messages are required.
Announcing over the DHT
Also, datasets are now announced by their Merkle root instead of each individual block as was the case in the previous implementation. Announcing individual blocks is still supported, for example manifests are announced exactly the same as before, by their cid. Announcing individual blocks is also supported (but not required) and can be usefull in the case of bandwidth incentives.
Block Stores and Local Repo
All interaction with blocks/chunks sit behind the BlockStore
abstraction, which currently only supports querying blocks by hash. It should be extended to allow querying by Merkle root and block index and/or range.
The local repo should be aware of the persisted Merkle tree. When a requests by index is made, the store first locates the persisted Merkle tree corresponding to the specified root and retrieves the requested leaf and corresponding Merkle proofs.
Once the hash of the request block is known, the repo/store can be queried for the block using the retrieved block hash.
Keeping support for hash based retrieval (content addressing) has two main advantages:
- It preserves content addressing at the repo level, which enables content deduplication.
- It allows keeping the number of required changes to a minimum, as once the block hash is know, the existing flow can be reused.
Updated flow
Upload
sequenceDiagram
User ->> +Client: Upload file
loop Store Block
Client ->> +Chunker: Data Stream
loop Chunking
Chunker -->> +Chunker: Chunk stream
Chunker -->> -Client: Block
end
Client ->> +Repo: Put block
Client ->> +MerkleTree: Add Block Hash
end
MerkleTree -->> -Client: Merkle Root
Client ->> MerkleTree: Serialize Merkle Tree
Client ->> Client: Put Merkle Root in Manifest
Client ->> Repo: Persist Manifest
Client -->> User: Manifest Cid
Client ->> DHT: Announce Manifest Cid
Client ->> -DHT: Announce Dataset Cid
Steps:
- User initiates a file upload
- Client chunks the stream and stores blocks in the Repo
- Block's hash is added to a MerkleTree instance
- This is repeated until all data has been read from the stream
- Once all blocks have been stored, the Merkle root is generated and persisted
- The manifest is persisted and serialized in the repo
- The cid of the persisted manifest is returned to the user
- Both the manifest Cid and the Dataset Merkle Root Cid are announced on the DHT
- This allows locating both the manifest and the dataset individually
Retrieval
Local Flow
sequenceDiagram
User ->> Client: Request Manifest Cid
alt If manifest cid in Repo
Client ->> Repo: getBlock(cid)
else Manifest cid not in Repo, request from Network
Client ->> NetworkStore: [See Network Flow]
end
Repo -->> Client: Manifest Block
Client ->> Client: Deserialize Manifest and Read Merkle Root
Client ->> MerkleTree: loadMerkleTree(manifest.cid)
loop Read Dataset
Client ->> MerkleTree: getLeaf(index)
MerkleTree -->> Client: [leaf cid, proof hashes...]
alt If cid in Repo
Client ->> Repo: getBlock(cid)
Repo -->> Client: Data Block
Client -->> User: Stream of blocks
else Cid not in Repo, request from Network
Client ->> NetworkStore: [See Network Flow]
end
end
Steps:
- User initiates a download with a manifest Cid
- Client checks the local store for the manifest Cid
- If it exists, the manifest is deserialized and the Merkle root of the dataset is read
- Otherwise, the Cid is requested from the network store
- Client checks the local repo for the Merkle tree root
- If it exists, the Merkle tree is deserialized and leaf hashes are read
- For each leaf hash which corresponds to the hash of the block
- The local repo is checked for the precense of the block
- If it exists, it is read from the local store and returned to the client
- Otherwise, the Cid is requested from the network store
- The local repo is checked for the precense of the block
Network Flow
sequenceDiagram
alt If block cid in Repo
Client ->> Repo: getBlock(cid)
Repo -->> Client: Block
else Not in repo or no cid for block
Client ->> NetworkStore: getBlockByIndex(cid, index)
NetworkStore ->> BlockExchange: requestBlock(cid, index)
loop Retrieve Blocks
alt If have peers for Cid
BlockExchange ->> Peers: Request root cid and index (or range)
break Found Block(s)
Peers -->> BlockExchange: [[block, [leaf cid, proof hashes...]]...]
end
else No peers for Cid
loop Find Peers
BlockExchange ->> DHT: Find peers for cid
break Peers Found
DHT -->> BlockExchange: [peers...]
end
end
end
end
BlockExchange -->> NetworkStore: [[block, [proof hashes...]]...]
loop For all blocks
alt If Block hash and Merkle proof is correct
NetworkStore -->> MerkleTree: Store Merkle path
NetworkStore -->> Repo: Store Block
NetworkStore -->> Client: Block
else Block hash and Merkle proof is incorrect
break Incorrect Block or Merkle proof
Client -> NetworkStore: Disconnect bad peer
end
end
end
end
Steps:
- The client requests blocks from the network store, using the Merkle root and block index
- Network store requests the block from the BlockExchange engine
- BlockExchange checks connected peers for requested hash
- If they do, the block is requested using the root hash and index (or range) of the block
- Otherwise, it queries the DHT for the requested root hash
- Once new peers have been discovered and connected, go to step 1.1.1
- BlockExchange checks connected peers for requested hash
- Network store requests the block from the BlockExchange engine
- Once blocks are received from the remote nodes
- The hashes are verified against the requested Merkle root and if they pass
- The block is persisted to the network
- The block hash (cid) and the Merkle proof are stored in the persisted Merkle tree
- Otherwise, the block is discarded and the node that sent the incorrect block disconnected
- The hashes are verified against the requested Merkle root and if they pass