diff --git a/swarm/docs/sw^3/bandwidth.rst b/swarm/docs/sw^3/bandwidth.rst index 7531d5d52800..27d003c3e176 100644 --- a/swarm/docs/sw^3/bandwidth.rst +++ b/swarm/docs/sw^3/bandwidth.rst @@ -145,5 +145,5 @@ By default nodes will store all chunks forwarded as the response to a retrieve r These lookup results are worth storing because repeated requests for the same chunk can be served from the node's local storage without the need to "purchase" the chunk again from others. This strategy implicitly takes care of auto-scaling the network. Chunks originating from retrieval traffic will fill up the local storage adjusting redundancy to use maximum dedicated disk/memory capacity of all nodes. A preference to store frequently retrieved chunks results in higher redundancy aligning with more current usage. All else being equal, the more redundant a chunk, the fewer forwarding hops are expected for their retrieval, thereby reducing expected latency as well as network traffic for popular content. -Whereas retrieval compensation may prove sufficient for keeping the network in a relatively healthy state in terms of latency, from a resilience point of view, extra incentives are needed. We turn to this problem now. +Whereas retrieval compensation may prove sufficient for keeping the network in a relatively healthy state in terms of latency, from a resilience point of view, more work is needed. We may need additional redundancy to be resilient against partial network outages and we need extra incentives to ensure long-term availablilty of content even when it is accessed rarely. In the following two sections we discuss these problems in turn. diff --git a/swarm/docs/sw^3/erasure.rst b/swarm/docs/sw^3/erasure.rst new file mode 100644 index 000000000000..3f2502133b07 --- /dev/null +++ b/swarm/docs/sw^3/erasure.rst @@ -0,0 +1,140 @@ + +Loss-tolerant Merkle Trees and Erasure Codes +------------------------------------------------- + +Recall that the basic data structure in swarm is that of a tree with 32 bytes at the nodes and 128 children per node. Each node represents the root hash of a subtree or, at the last level, the hash of a 4096 byte span (one chunk) of the file. Generically we may think of each chunk as consisting of 128 hashes: + +.. _fig:chunk: + +.. figure:: fig/chunk.pdf + :align: center + :alt: A chunk consisting of 128 hashes + :figclass: align-center + + A chunk consists of 4096 bytes of the file or a sequence of 128 subtree hashes. + +while in the tree structure, the 32 bytes stored at the node represent the hash of the 128 children. + +.. _fig:treebasic: + +.. figure:: fig/treebasic.pdf + :align: center + :alt: a generic node in the tree has 128 children. + :figclass: align-center + + A generic node in the tree has 128 childern. + +Recall also that during normal swarm lookups, a swarm client performs a lookup for a hash value and receives a chunk in return. This chunk in turn constitutes another 128 hashes to be looked up in return for another 128 hashes and so on until the chunks received belong to the actual file. Here is a schematic: (:numref:`Figure %s `): + +.. _fig:tree2: + +.. figure:: fig/tree2.pdf + :align: center + :alt: the swarm tree + :figclass: align-center + + The swarm tree broken into chunks. + + +Erasure coding the Swarm Tree +=================================== + +We propose using a Cauchy-Reed-Solomon (henceforth CRS) scheme to encode redundancy directly into the swarm tree. The CRS scheme is a systemic erasure code capable of implementing a scheme whereby any :math:`m` out of :math:`n` fix-sized pieces are sufficient to reconstruct the original data blob of size :math:`m` pieces with storage overhead of :math:`n-m` [#]_ . + +.. rubric:: Footnotes +.. [#] There are open source libraries to do Reed Solomon or Cauchy-Reed-Solomon coding. See https://www.usenix.org/legacy/event/fast09/tech/full_papers/plank/plank_html/, https://www.backblaze.com/blog/reed-solomon/, http://rscode.sourceforge.net/. + +Once we have the :math:`m` pieces of the original blob, CRS scheme provides a method to inflate it to size :math:`n` by supplementing :math:`n-m` so called parity pieces. With that done, assuming :math:`p` is the probability of losing one piece, if all :math:`n` pieces are independently stored, the probability of loosing the original content is :math:`p^{n-m+1}` exponential while extra storage is linear. These properties are preserved if we apply the coding to every level of a swarm chunk tree. + +Assuming we fix :math:`n=128` the branching factor of the swarm hash (chunker). +The chunker algorithm would proceed the following way when splitting the document: + + 1. Set input to the data blob. + 2. Read the input 4096 byte chunks at a time. Count the chunks by incrementing :math:`i`. + IF fewer than 4096 bytes are left in the file, fill up the last fraction to 4096 + 3. Repeat 1 until there's no more data or :math:`i \equiv 0` mod :math:`m` + 4. If there is no more data add padding of :math:`j` chunks such that :math:`i+j \equiv 0` mod :math:`m`. + 5. use the CRS scheme on the last :math:`m` chunks to produce :math:`128-m` parity chunks resulting in a total of 128 chunks. + 6. Record the hashes of the 128 chunks concatenated to result in the next 4096 byte chunk of the next level. + 7. If there is more data repeat 1. otherwise + 8. If the next level data blob is of size larger than 4096, set the input to this and repeat from 1. + 9. Otherwise remember the blob as the root chunk + + +Let us now suppose that we divide our file into 100 equally sized pieces, and then add 28 +more parity check pieces using a Reed-Solomon code so that now any 100 of the 128 pieces are +sufficient to reconstruct the file. On the next level up the chunks are composed of the hashes of +their first hundered data chunks and the 28 hashes of the parity chunks. Let’s take the first 100 +of these and add an additional 28 parity chunks to those such that any 100 of the resulting 128 +chunks are sufficient to reconstruct the origial 100 chunks. And so on on every level. In terms of +availability, every subtree is equally important to every other subtree at this level. The resulting +data structure is not a balanced tree since on every level i the last 28 chunks are parity leaf +chunks while the first 100 are branching nodes encoding a subtree of depth :math:`i-1` redundantly. +Then a typical piece of our tree would look like this: (:numref:`Figure %s `) + +.. _fig:tree-with-erasure: + +.. figure:: fig/tree-with-erasure.pdf + :align: center + :alt: the swarm tree with erasure coding + :figclass: align-center + + The swarm tree with extra parity chunks. Chunks :math:`p^{101}` through :math:`p^{128}` are parity data for chunks :math:`h^1_1 - h^1_{128}` through :math:`h^{100}_1 - h^{100}_{128}`. + + + + +Two things to note + + * This pattern repeats itself all the way down the tree. Thus hashes :math:`h^1_{101}` through :math:`h^1_{128}` point to parity data for chunks pointed to by :math:`h^1_1` through :math:`h^1_{100}`. + * Parity chunks :math:`p^i` do not have children and so the tree structure does not have uniform depth. + +The special case of the last chunks in each row +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +If the size of the file is not a multiple of 4096 bytes, then it cannot be evenly divided into chunks. The current strategy is to simply fill up the last incomplete chunk to fill 4096 bytes. The next step in the encoding process is to add the parity chunks. We choose some erasure coding redundancy parameters (for example 100-of-128) and we process the file 100 chunks at a time, encoding to 128 each time. + +If the number of file chunks is not divisible by 100, then we cannot proceed with the last batch in the same way as the others. We propose that we encode the remaining chunks with an erasure code that guarantees at least the same level of security as the others. Note that it is not as simple as choosing the same redundancy. For example a 50-of-100 encoding is much more secure against file loss than a 1-of-2 encoding even though the redundancy is 100% in both cases. Overcompensating, we could say that there should always be the same number of parity chunks (eg. 28) even when there are fewer (than 100) data chunks so that we alwasy end up with m-of-m+28. We repeat this procedure in every row in the tree. + +However it is not possible to use our m-of-n scheme on a single chunk (m=1) because it would amount to n copies of the same chunk. The problem of course is that any number of copies of the same chunk all have the same hash and are therefore indistinguishable in the swarm. Thus when there is only a single chunk left over at some level of the tree, we'd have to apply some transformation to it to obtain a second (but different) copy before we could generate more parity chunks. + +In particular this is always the case for the root chunk. To illustrate why this is critically important, consider the following. The root hash points to the root chunk. If this root chunk is lost, then the file is not retrievable from the swarm even if all other data is present. Thus we must find an additional method of securing and accessing the information stored in the root chunk. + +Whenever a single chunk is left over (m=1) we propose the following procedure. + + 1. If the chunk is smaller than 4096 bytes, we use diferential padding to make n different 4096-byte chunks containing the data. + 2. If the chunk is full size we apply n different reversible permutations to get n different copies. For example we could use cyclic permutations [#]_ . + +.. rubric:: Footnotes +.. [#] Alternatively, we could formally differentiate the chunks using the filesize data. In swarm, each 4096 byte chunk is actually stored together with 8 bytes of meta information - currently only storing the size of the subtree encoded by the chunk. It is plausible for a future implementation of swarm to use 1 byte of meta-information in order to differentiate multiple copies of the same data. + + +Benefits of CRS merkle tree +============================= + +This per-level m-of-n Cauchy-Reed-Solomon erasure code introduced into the swarm chunk tree does not only ensure file availability, but also offers further benefits of increased resilience and ways to speed up retrieval. + +All chunks are created equal +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +A tree encoded as suggested above has the same redundancy at every node [#]_. + +.. rubric:: Footnotes +.. [#] If the filesize is not a specific multiple of 4096 bytes, then the last chunk at every level will actually have a higher redundancy even than the rest. + +This means that chunks nearer to the root are no longer more important than chunks near the file. Every node has an m-of-128 redundancy level and no chunk after the root chunk is more important than any other chunk. +A problem that immediately presents itself is the following: if nodes are compensated only for serving chunks, then less popular chunks are less profitable and more likely to be deleted; therefore, if users only download the 100 data chunks and never request the parity chunks, then these are more likely to get deleted and ultimately not be available when they are finally needed. +Another approach would be to use non-systemic coding. A systemic code is one in which the data remains intact and we add extra parity data whereas in a non-systemic code we replace all data with parity data such that (in our example) all 128 pieces are really created equal. While the symmetry of this approach is appealing, this leads to forced decoding and thus to a high CPU usage even in normal operation and it also prevents us from streaming files from the swarm. +Luckily the problem is solved by the automated audit scheme which audits the integrity of all chunks and does not distinguish between data or parity chunks. + +Self healing +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Any client downloading a file from the swarm can detect if a chunk has been lost. The client can reconstruct the file from the parity data (or reconstruct the parity data from the file) and resync this data into the swarm. That way, even if a large fraction of the swarm is wiped out simultaneously, this process should allow an organic healing process to occur and it is encouraged that the default client behavior should be to repair any damage detected. + +Improving latecy of retrievals +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Alpha is the name the original Kademlia gives to the number of peers in a Kademlia bin that are queried simultaneously during a lookup. The original Kademlia paper sets alpha=3. This is impractical for Swarm because the peers do not report back with new addresses as they would do in pure Kademlia but instead forward all queries to their peers. Swarm is coded in this way to make use of semi-stable longer-term devp2p connections. Setting alpha to anything greater than 1 thus increases the amount of network traffic substantially – setting up an exponential cascade of forwarded lookups (but it would soon collapse back down onto the target of the lookup). +However, setting alpha=1 has its own downsides. For instance, lookups can stall if they are forwarded to a dead node and even if all nodes are live, there could be large delays before a query is complete. The practice of setting alpha=2 in swarm is designed to speed up file retrieval and clients are configured to accept chunks from the first/fastest forwarding connection to be established. +In an erasure coded setting we can in a sense have a best of both worlds. The default behavior should be to set alpha=1 i.e. to query one peer only for each chunk lookup, but crucially, to issue a lookup request not just for the data chunks but for the parity chunks as well. The client then could accept the first m of every 128 chunks queried to get some of the same benefits of faster retrieval that redundant lookups provide without a whole exponential cascade. + diff --git a/swarm/docs/sw^3/fig/chunk.pdf b/swarm/docs/sw^3/fig/chunk.pdf new file mode 100644 index 000000000000..52e6f722dbcb Binary files /dev/null and b/swarm/docs/sw^3/fig/chunk.pdf differ diff --git a/swarm/docs/sw^3/fig/chunk.tex b/swarm/docs/sw^3/fig/chunk.tex new file mode 100644 index 000000000000..5026cf9ec615 --- /dev/null +++ b/swarm/docs/sw^3/fig/chunk.tex @@ -0,0 +1,20 @@ +\documentclass{standalone} +%\documentclass[a4paper,10pt]{scrartcl} + +\usepackage[utf8]{inputenc} +\usepackage{tikz} +\usetikzlibrary{fit} + +\begin{document} + +\begin{tikzpicture} +\node at (0,1) {Chunk:}; +\node[draw,dashed] (h1) at (2,1) {$h_1$}; +\node[draw,dashed] (h2) at (3,1) {$h_2$}; +\node[draw,dashed] (h3) at (4,1) {$h_3$}; +\node (dots) at (6,1) {$\cdots$}; +\node[draw,dashed] (h128) at (8,1) {$h_{128}$}; +\node[draw,fit=(h1) (h2) (h3) (dots) (h128)]{}; +\end{tikzpicture} + +\end{document} diff --git a/swarm/docs/sw^3/fig/tree-with-erasure.pdf b/swarm/docs/sw^3/fig/tree-with-erasure.pdf new file mode 100644 index 000000000000..9f37f8ab2bc5 Binary files /dev/null and b/swarm/docs/sw^3/fig/tree-with-erasure.pdf differ diff --git a/swarm/docs/sw^3/fig/tree-with-erasure.tex b/swarm/docs/sw^3/fig/tree-with-erasure.tex new file mode 100644 index 000000000000..dafbd9593f2d --- /dev/null +++ b/swarm/docs/sw^3/fig/tree-with-erasure.tex @@ -0,0 +1,35 @@ +\documentclass{standalone} +%\documentclass[a4paper,10pt]{scrartcl} + +\usepackage[utf8]{inputenc} +\usepackage{tikz} +\usetikzlibrary{fit} + +\begin{document} + +\begin{tikzpicture} +\node[draw,dashed] (root) at (3.5,3) {hash($h_1$-$h_{128}$)}; +\node[draw,dashed] (h1) at (1,1) {$h_1$}; +\node (dots1) at (2,1) {$\cdots$}; +\node[draw,dashed] (h100) at (3,1) {$h_{100}$}; +\node[draw,fill=lightgray,dashed] (h101) at (4,1) {$h_{101}$}; +\node (dots2) at (5,1) {$\cdots$}; +\node[draw,fill=lightgray,dashed] (h128) at (6,1) {$h_{128}$}; +\node[draw,fit=(h1) (dots1) (h100) (h101) (dots2) (h128)]{}; +\draw (root) -- (h1); +\draw (root) -- (h100); +\draw (root) -- (h101); +\draw (root) -- (h128); +% +\node[draw] (g1) at (0.5,-1) {$h^1_1$ - $h^1_{128}$}; +\node[draw] (g100) at (2.5,-1) {$h^{100}_1$ - $h^{100}_{128}$}; +\node[draw,fill=gray] (p101) at (4,-1) {$p^{101}$}; +\node[draw,fill=gray] (p128) at (6,-1) {$p^{128}$}; +% +\draw (h1) -- (g1); +\draw (h100) -- (g100.30); +\draw (h101) -- (p101); +\draw (h128) -- (p128); + +\end{tikzpicture} +\end{document} diff --git a/swarm/docs/sw^3/fig/tree2.pdf b/swarm/docs/sw^3/fig/tree2.pdf new file mode 100644 index 000000000000..10037b3f305b Binary files /dev/null and b/swarm/docs/sw^3/fig/tree2.pdf differ diff --git a/swarm/docs/sw^3/fig/tree2.tex b/swarm/docs/sw^3/fig/tree2.tex new file mode 100644 index 000000000000..148f7acebb95 --- /dev/null +++ b/swarm/docs/sw^3/fig/tree2.tex @@ -0,0 +1,49 @@ +\documentclass{standalone} +%\documentclass[a4paper,10pt]{scrartcl} + +\usepackage[utf8]{inputenc} +\usepackage{tikz} +\usetikzlibrary{fit} + +\begin{document} + +\begin{tikzpicture} +\node[draw,dashed] (root) at (5,3) {root hash}; +\node[draw,dashed] (h1) at (1,1) {$h_1$}; +\node[draw,dashed] (h2) at (2,1) {$h_2$}; +\node[draw,dashed] (h3) at (3,1) {$h_3$}; +\node (dots) at (5,1) {$\cdots$}; +\node[draw,dashed] (h128) at (7,1) {$h_{128}$}; +\node[draw,fit=(h1) (h2) (h3) (dots) (h128)]{}; +\draw (root) -- (h1); +\draw (root) -- (h2); +\draw (root) -- (h3); +\draw (root) -- (h128); +\node[draw,dashed] (g1) at (-1.2,-1) {$h^1_1$}; +\node[draw,dashed] (g2) at (-0.5,-1) {$h^1_2$}; +\node (gdots) at (0,-1) {$\cdots$}; +\node[draw,dashed] (g128) at (0.7,-1) {$h^1_{128}$}; +\draw (h1) -- (g1); +\draw (h1) -- (g2); +\draw (h1) -- (g128); +\node[draw,fit=(g1)(g2)(gdots)(g128)]{}; +\node[draw,dashed] (f1) at (1.8,-1) {$h^2_1$}; +\node[draw,dashed] (f2) at (2.5,-1) {$h^2_2$}; +\node (fdots) at (3,-1) {$\cdots$}; +\node[draw,dashed] (f128) at (3.7,-1) {$h^2_{128}$}; +\draw (h2) -- (f1); +\draw (h2) -- (f2); +\draw (h2) -- (f128); +\node[draw,fit=(f1)(f2)(fdots)(f128)]{}; +\node (moredots) at (5,-1) {$\cdots$}; +\node[draw] (c1) at (-1.5,-3) {chunk 1}; +\node[draw] (c2) at (0,-3) {chunk 2}; +\node at (1.5,-3) {$\cdots$}; +\node[draw] (c129) at (3,-3) {chunk 129}; +\node (cdots) at (4.5,-3) {$\cdots$}; +\node[draw] (cn) at (6,-3) {chunk N}; +\draw (g1) -- (c1); +\draw (g2) -- (c2); +\draw (f1) -- (c129); +\end{tikzpicture}\\ +\end{document} diff --git a/swarm/docs/sw^3/fig/treebasic.pdf b/swarm/docs/sw^3/fig/treebasic.pdf new file mode 100644 index 000000000000..c76bf14fcc0c Binary files /dev/null and b/swarm/docs/sw^3/fig/treebasic.pdf differ diff --git a/swarm/docs/sw^3/fig/treebasic.tex b/swarm/docs/sw^3/fig/treebasic.tex new file mode 100644 index 000000000000..c04733a54a11 --- /dev/null +++ b/swarm/docs/sw^3/fig/treebasic.tex @@ -0,0 +1,23 @@ +\documentclass{standalone} +%\documentclass[a4paper,10pt]{scrartcl} + +\usepackage[utf8]{inputenc} +\usepackage{tikz} +\usetikzlibrary{fit} + +\begin{document} + +\begin{tikzpicture} +\node[draw,dashed] (root) at (5,3) {hash of chunk $h_1$ - $h_{128}$}; +\node[draw,dashed] (h1) at (1,1) {$h_1$}; +\node[draw,dashed] (h2) at (2,1) {$h_2$}; +\node[draw,dashed] (h3) at (3,1) {$h_3$}; +\node (dots) at (5,1) {$\cdots$}; +\node[draw,dashed] (h128) at (7,1) {$h_{128}$}; +% \node[draw,fit=(h1) (h2) (h3) (dots) (h128)]{}; +\draw (root) -- (h1); +\draw (root) -- (h2); +\draw (root) -- (h3); +\draw (root) -- (h128); +\end{tikzpicture} +\end{document} diff --git a/swarm/docs/sw^3/index.rst b/swarm/docs/sw^3/index.rst index d9dc4fb8abc1..e73882baf8bd 100644 --- a/swarm/docs/sw^3/index.rst +++ b/swarm/docs/sw^3/index.rst @@ -16,6 +16,7 @@ Welcome to the swarm documentation! :maxdepth: 4 bandwidth + erasure storage swap parameters diff --git a/swarm/docs/sw^3/storage.rst b/swarm/docs/sw^3/storage.rst index e87807cd30a0..4aee4df8a01b 100644 --- a/swarm/docs/sw^3/storage.rst +++ b/swarm/docs/sw^3/storage.rst @@ -115,76 +115,7 @@ Assuming all chunks of the original file are different this yields a potential .. rubric:: Footnotes .. [#] We also explored the possibility that degree of redundancy is subsumed under local replication. Local replicas are instances of a chunk stored by nodes in a close neighbourhood. If that particular chunk is crucial in the reconstruction of the content, the swarm is much more vulnerable to chunk loss or latency due to attacks. This is because if the storers of the replicas are close, inflitrating in the storers' neighbourhood can be done with as many nodes as chunk type (as opposed to as many as chunk replicas). If there is cost to sybil attacks this brings down the cost by a factor of n where n is the number of replicas. We concluded that local replication is important for resilience in case of intermittend node dropouts, however, inferior to other solutions to express security level as expressed by the owner. -Luckily there are a lot more economical ways to encode a file redundantly. In what follows we spell out our proposal to introduce a scheme for loss tolerant swarm hash. - -Loss-tolerant Merkle Trees ----------------------------------------------------------- - -Recall that each node (except possibly the last one on each level) has 128 children each of which represent the root hash of a subtree or, at the last level, represent a 4096 byte span of the file. Let us now suppose that we divide our file into 100 equally sized pieces, and then add 28 more parity check pieces using a Reed-Solomon code so that now any 100 of the 128 pieces are sufficient to reconstruct the file. On the next level up the chunks are composed of the hashes of their first hundered data chunks and the 28 hashes of the parity chunks. Let's take the first 100 of these and add an additional 28 parity chunks to those such that any 100 of the resulting 128 chunks are sufficient to reconstruct the origial 100 chunks. And so on on every level. -In terms of availability, every subtree is equally important to every other subtree at this level. The resulting data structure is not a balanced tree since on every level :math:`i` the last 28 chunks are parity leaf chunks while the first 100 are branching nodes encoding a subtree of depth :math:`i-1` redundantly. - -In practice of course, data chunks are still prefered over the parity chunks in order to avoid CPU overhead in reconstruction. This data structure has preserved its merkle properties and can be used for partial integrity check. - - - -The Cauchy-Reed-Solomon (henceforth CRS) scheme is a systemic erasure code capable of implementing a scheme whereby any :math:`m` out of :math:`n` fix-sized pieces are able to reconstruct the original data blob of size :math:`m` pieces with storage overhead of -:math:`n-m` [#]_ . Once we got the :math:`m` pieces of the original blob, CRS scheme provides a method to inflate it to size :math:`n` by supplementing :math:`n-m` so called parity pieces. With that done, assuming `p` is the probability of losing one piece, if all :math:`n` pieces are independently stored, the probability of loosing the original content is :math:`p^{n-m+1}` exponential while extra storage is linear. These properties are preserved if we apply the coding to every level of a swarm chunk tree. - -.. rubric:: Footnotes -.. [#] There are open source libraries to do Reed Solomon or Cauchy-Reed-Solomon coding. See https://www.usenix.org/legacy/event/fast09/tech/full_papers/plank/plank_html/, https://www.backblaze.com/blog/reed-solomon/, http://rscode.sourceforge.net/. - -Assuming we fix :math:`n=128` the branching factor of the swarm hash (chunker). -The chunker algorithm would proceed the following way when splitting the document: - -0. Set input to the data blob. -1. Read the input 4096 byte chunks at a time. Count the chunks by incrementing :math:`i` - IF fewer than 4096 bytes are left in the file, fill up the last fraction to 4096 -2. Repeat 1 until there's no more data or :math:`i \mod m=0` -3. If there is no more data add padding of :math:`j` chunks such that :math:`i+j \mod m=0`. -3. use the CRS scheme on the last :math:`m` chunks to produce :math:`128-m` parity chunks resulting in a total of 128 chunks. -4. Record the hashes of the 128 chunks concatenated to result in the next 4096 byte chunk of the next level. -5. If there is more data repeat 1. otherwise -6. If the next level data blob is of size larger than 4096, set the input to this and repeat from 1. -7. Otherwise remember the blob as the root chunk - - -Benefits of CRS merkle tree -------------------------------------- - -This per-level m-of-n Cauchy-Reed-Solomon erasure code introduced into the swarm chunk tree does not only ensure file availability, but also offers further benefits of increased resilience and ways to speed up retrieval. - - -All chunks are created equal -^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ - -A tree encoded as suggested above has the same redundancy at every node [#]_ . This means that chunks nearer to the root are no longer more important than chunks near the file. Every node has an m-of-128 redundancy level and no chunk after the root chunk is more important than any other chunk. - -.. rubric:: Footnotes -.. [#] If the filesize is not a specific multiple of 4096 bytes, then the last chunk at every level will actually have a higher redundancy even than the rest. - - -A problem that immediately presents itself is the following: if nodes are compensated only for serving chunks, then less popular chunks are less profitable and more likely to be deleted; therefore, if users only download the 100 data chunks and never request the parity chunks, then these are more likely to get deleted and ultimately not be available when they are finally needed. - -Another approach would be to use non-systemic coding. A systemic code is one in which the data remains intact and we add extra parity data whereas in a non-systemic code we replace all data with parity data such that (in our example) all 128 pieces are really created equal. While the symmetry of this approach is appealing, this leads to forced decoding and thus to a high CPU usage even in normal operation and it also prevents us from streaming files from the swarm. - -Luckily the problem is solved by the automated audit scheme which audits the integrity of all chunks and does not distinguish between data or parity chunks. - -Self healing -^^^^^^^^^^^^^^^^^^^^^^ - -Any client downloading a file from the swarm can detect if a chunk has been lost. The client can reconstruct the file from the parity data (or reconstruct the parity data from the file) and resync this data into the swarm. That way, even if a large fraction of the swarm is wiped out simultaneously, this process should allow an organic healing process to occur and it is encouraged that the default client behavior should be to repair any damage detected. - -Improving latecy of retrievals -^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ - -Alpha is the name the original Kademlia gives to the number of peers in a Kademlia bin that are queried simultaneously during a lookup. The original Kademlia paper sets alpha=3. This is impractical for Swarm because the peers do not report back with new addresses as they would do in pure Kademlia but instead forward all queries to their peers. Swarm is coded in this way to make use of semi-stable longer-term devp2p connections. Setting alpha to anything greater than 1 thus increases the amount of network traffic substantially – setting up an exponential cascade of forwarded lookups (but it would soon collapse back down onto the target of the lookup). -However, setting alpha=1 has its own downsides. For instance, lookups can stall if they are forwarded to a dead node and even if all nodes are live, there could be large delays before a query is complete. The practice of setting alpha=2 in swarm is designed to speed up file retrieval and clients are configured to accept chunks from the first/fastest forwarding connection to be established. -In an erasure coded setting we can in a sense have a best of both worlds. The default behavior should be to set alpha=1 i.e. to query one peer only for each chunk lookup, but crucially, to issue a lookup request not just for the data chunks but for the parity chunks as well. The client then could accept the first m of every 128 chunks queried to get some of the same benefits of faster retrieval that redundant lookups provide without a whole exponential cascade. - - -Improving resilience in case of non-saturated Kademlia table -^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ - +Luckily there are a lot more economical ways to encode a file redundantly. In particular the erasure coded, loss tolerant merkle tree, discussed in the previous section, allows the user to choose their own level of guaranteed data redundancy and security. From here on we assume that the user applied CRS encoding when splitting their content and therefore expressed their desired degree of redundancy in the CRS parameters, the price of which they pay in terms of the increased number of chunks they need to pay storage for without adding complexity to the storage distribution and pricing.