diff --git a/content/.gitbook/assets/chunk-distribution-dynamic-pattern.png b/content/.gitbook/assets/chunk-distribution-dynamic-pattern.png
new file mode 100644
index 000000000..06fd1ba97
Binary files /dev/null and b/content/.gitbook/assets/chunk-distribution-dynamic-pattern.png differ
diff --git a/content/.gitbook/assets/chunk-distribution-static-pattern.png b/content/.gitbook/assets/chunk-distribution-static-pattern.png
new file mode 100644
index 000000000..a888c51c9
Binary files /dev/null and b/content/.gitbook/assets/chunk-distribution-static-pattern.png differ
diff --git a/content/.gitbook/assets/prolly-tree-back-insert.png b/content/.gitbook/assets/prolly-tree-back-insert.png
new file mode 100644
index 000000000..8b84f887f
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-back-insert.png differ
diff --git a/content/.gitbook/assets/prolly-tree-delete.png b/content/.gitbook/assets/prolly-tree-delete.png
new file mode 100644
index 000000000..9b73e7b2d
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-delete.png differ
diff --git a/content/.gitbook/assets/prolly-tree-diff-step-1.png b/content/.gitbook/assets/prolly-tree-diff-step-1.png
new file mode 100644
index 000000000..f86e171d3
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-diff-step-1.png differ
diff --git a/content/.gitbook/assets/prolly-tree-diff-step-2.png b/content/.gitbook/assets/prolly-tree-diff-step-2.png
new file mode 100644
index 000000000..23a42cef5
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-diff-step-2.png differ
diff --git a/content/.gitbook/assets/prolly-tree-diff-step-3.png b/content/.gitbook/assets/prolly-tree-diff-step-3.png
new file mode 100644
index 000000000..567d8280c
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-diff-step-3.png differ
diff --git a/content/.gitbook/assets/prolly-tree-diff-step-4.png b/content/.gitbook/assets/prolly-tree-diff-step-4.png
new file mode 100644
index 000000000..17f78ee7c
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-diff-step-4.png differ
diff --git a/content/.gitbook/assets/prolly-tree-front-insert.png b/content/.gitbook/assets/prolly-tree-front-insert.png
new file mode 100644
index 000000000..9837801a2
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-front-insert.png differ
diff --git a/content/.gitbook/assets/prolly-tree-history-independence-1.png b/content/.gitbook/assets/prolly-tree-history-independence-1.png
new file mode 100644
index 000000000..20a923f7f
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-history-independence-1.png differ
diff --git a/content/.gitbook/assets/prolly-tree-history-independence-2.png b/content/.gitbook/assets/prolly-tree-history-independence-2.png
new file mode 100644
index 000000000..70264ffa6
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-history-independence-2.png differ
diff --git a/content/.gitbook/assets/prolly-tree-middle-insert.png b/content/.gitbook/assets/prolly-tree-middle-insert.png
new file mode 100644
index 000000000..46a4d8d80
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-middle-insert.png differ
diff --git a/content/.gitbook/assets/prolly-tree-update-value.png b/content/.gitbook/assets/prolly-tree-update-value.png
new file mode 100644
index 000000000..7f29eeff5
Binary files /dev/null and b/content/.gitbook/assets/prolly-tree-update-value.png differ
diff --git a/content/.gitbook/assets/tim-prolly-tree-step-1.png b/content/.gitbook/assets/tim-prolly-tree-step-1.png
new file mode 100644
index 000000000..00500dbc4
Binary files /dev/null and b/content/.gitbook/assets/tim-prolly-tree-step-1.png differ
diff --git a/content/.gitbook/assets/tim-prolly-tree-step-2.png b/content/.gitbook/assets/tim-prolly-tree-step-2.png
new file mode 100644
index 000000000..f25361fe8
Binary files /dev/null and b/content/.gitbook/assets/tim-prolly-tree-step-2.png differ
diff --git a/content/.gitbook/assets/tim-prolly-tree-step-3.png b/content/.gitbook/assets/tim-prolly-tree-step-3.png
new file mode 100644
index 000000000..d9727102a
Binary files /dev/null and b/content/.gitbook/assets/tim-prolly-tree-step-3.png differ
diff --git a/content/.gitbook/assets/tim-prolly-tree-step-5.png b/content/.gitbook/assets/tim-prolly-tree-step-5.png
new file mode 100644
index 000000000..7b2e2c4e4
Binary files /dev/null and b/content/.gitbook/assets/tim-prolly-tree-step-5.png differ
diff --git a/content/architecture/storage-engine/prolly-tree.md b/content/architecture/storage-engine/prolly-tree.md
index bd57115c4..d0996c462 100644
--- a/content/architecture/storage-engine/prolly-tree.md
+++ b/content/architecture/storage-engine/prolly-tree.md
@@ -2,38 +2,238 @@
title: Prolly Tree
---
-# Prolly Tree
+"Prolly Tree" is short for ["Probabilistic B-tree"](https://github.com/attic-labs/noms/blob/master/doc/intro.md#prolly-trees-probabilistic-b-trees). "Prolly Tree" was coined by the good folks who built [Noms](https://github.com/attic-labs/noms), who as far as we can tell invented the data structure. We here at [DoltHub](https://www.dolthub.com) have immense respect for their pioneering work, without which [Dolt](https://www.doltdb.com) would not exist.
-In this section we explore how Dolt uses Prolly Trees.
+![Prolly Tree](../../.gitbook/assets/tim-prolly-tree-example.png)
-We draw on a [blog post](https://www.dolthub.com/blog/2020-04-01-how-dolt-stores-table-data/) we published about Prolly Trees, and an excellent [write up](https://github.com/attic-labs/noms/blob/master/doc/intro.md#prolly-trees-probabilistic-b-trees) by the Noms team.
+A Prolly Tree is a data structure closely related to a [B-tree](https://en.wikipedia.org/wiki/B-tree). Prolly Trees are generally useful but have proven particularly effective as the basis of [the storage engine](https://docs.dolthub.com/architecture/storage-engine) for [version controlled databases](https://www.dolthub.com/blog/2022-08-04-database-versioning/).
-One key data structure that Noms employs is the Prolly Tree, used for storing collections. A Prolly Tree combines ideas from [B-Trees](https://en.wikipedia.org/wiki/B-tree) and [Merkle Trees](https://en.wikipedia.org/wiki/Merkle_tree).
+# Motivation
-B-Trees are block oriented data structures with sorted key-value pairs stored at leaf nodes. Internal nodes store pointers to child nodes and ranges of keys which those child nodes provide access to. Reading from a B-Tree index for a given set of keys involves locating a range in which the desired key is located and following the pointers down the tree until arriving at the leaf node that stores the desired value. Prolly Trees resemble B-Trees in that they store all key-value data in leaf nodes, exactly like a B-tree, and internal nodes with key delimiters and pointers to their children, exactly like a B-tree.
+Let's say you need a data structure with the following properties:
-The point of difference is where Prolly Trees draw on ideas from Merkle Trees, widely used in distributed protocols such as Bitcoin and IPFS. Instead of internal nodes storing pointers to child nodes, and leaf nodes storing key value pairs, both the leaf nodes and the internal nodes are variable-length and content-addressed. The child pointers of the internal nodes are the content addresses of the leaf or internal nodes that they point to. This means the values themselves can be stored in an immutable content-addressed chunk store, and the content addresses used as the "pointers" in the Prolly Tree.
+1. **B-tree like performance** - Specifically on reads and writes.
+2. **Fast Diffs** - The ability to compare two versions efficiently.
+3. **Structural Sharing** - Any given portion of the data shared between versions is only stored once.
-When a value in a Prolly Tree changes, we only write the new value to a content addressed chunk, and recompute the nodes that lead to that chunk. The rest of the tree remains unchanged. The result is a storage footprint of new versions proportional to the size of the changes in the new version. This is called structural sharing.
+A Prolly Tree delivers these properties.
+
+Operation | B-Trees | Prolly Trees
+--------- | ------- | ------------
+1 Random Read | logk(n) | logk(n)
+1 Random Write | logk(n) | (1+k/w)*logk(n)
+Ordered scan of one item with size z | z/k | z/k
+Calculate diff of size d | n | d
+Structural sharing | ❌ | ✅
+
+**n**: total leaf data in tree, **k**: average block size, **w**: window width
+
+As you can see a Prolly Tree approximates B-tree performance on reads and writes while also offering the ability to compute differences in time proportional to the size of differences rather than the total size of the data. Prolly Trees can also structurally share portions of the tree between versions due to the content-addressed nature of their intermediary storage.
+
+Let's dive a bit deeper and show you how.
+
+# How Prolly Trees work
+
+Prolly Trees are a variant of B-trees so let's first review some key B-tree concepts and then dive into how Prolly Trees work in light of those concepts.
+
+## B-tree Review
+
+A B-tree is data structure that maps keys to values. A B-tree stores key-value pairs in leaf nodes in sorted order. Internal nodes of a B-tree store pointers to children nodes and key delimiters; everything reachable from a pointer to a child node falls within a range of key values corresponding to the key delimiters before and after that pointer within the internal node.
+
+![B-tree Example](../../.gitbook/assets/tim-b-tree-example.png)
+
+A B-tree is optimized for a tradeoff between write and read performance. It's more expensive to maintain than dropping tuples into a heap without any ordering constraints, but when data is stored in a B-tree it's much quicker to seek to a given key and to do an in-order traversal of the keys with their values.
+
+However, finding the differences between two B-trees requires scanning both trees and comparing all values, an operation that scales with the size of the tree. This operation is slow for large trees.
+
+Also, writes to B-trees are not history independent, the order of the writes internally changes the structure of the tree. Thus, storage cannot be easily shared between two versions of the same tree.
+
+## Building a Prolly Tree
+
+The easiest way to understand Prolly trees is to walk through, step-by-step, how we build one. You start with a map of keys to values. Using the above B-tree example, we have keys between 1 and 21. The values don't really matter. Just imagine them dangling off the keys.
+
+1. **Sort**: Sort the map by its key value, so that it is laid out in order.
+
+![Building a Prolly-Tree, Step 1](../../.gitbook/assets/tim-prolly-tree-step-1.png)
+
+2. **Determine Chunk Boundaries**: Use a seed, the size of the current chunk, the key value and a strong hash function to calculate whether this current entry represents a new chunk boundary. Any time the hash value is below a target value, form a chunk boundary and start a new block. Here's the chunking step on our leaf nodes:
+
+![Building a Prolly-tree, Step 2](../../.gitbook/assets/tim-prolly-tree-step-2.png)
+
+3. **Hash each Chunk**: You now have the leaf nodes of the Prolly Tree. Compute the content address of each block by applying a strong hash function to its contents. You store the contents in a content addressed block store. Here our blocks have been addressed and stored in the block store:
+
+![Building a Prolly-tree, Step 3](../../.gitbook/assets/tim-prolly-tree-step-3.png)
+
+4. **Finished?**: If the length of your list of content addresses is 1, then you are done. This is the content address of your tree.
+
+5. **Build a New Map** Otherwise, form the next layer of your Prolly Tree by creating a map of the highest key value in the chunk and the content address of the chunk. Here's what the entries for the first internal level of the tree would look like for our example:
+
+![Building a Prolly-tree, Step 5](../../.gitbook/assets/tim-prolly-tree-step-5.png)
+
+6. **Return to Step 2**: Use this new map step 2. The algorithm terminates when step 4's condition is reached.
+
+Following these steps results in the following Prolly Tree.
+
+![Prolly Tree](../../.gitbook/assets/tim-prolly-tree-example.png)
+
+## Modifying a Prolly Tree
+
+The magic of Prolly Trees is not seen on construction, but on modification. As you'll see, modifying a Prolly Tree changes the minimal amount of content addresses. Minimal modification means more structural sharing of contents because fewer content addresses change. It is also the basis of fast diff which is accomplished by comparing content addresses.
+
+### Update a Value
+
+If we update a value, we walk the tree by key to the leaf node holding the value. We then create the new leaf chunk by directly modifying the existing value in a copy of the existing chunk (ie. [copy on write](https://en.wikipedia.org/wiki/Copy-on-write)). After the edit, we recalculate the content address of the chunk. We then walk up the tree recalculating each internal content address up to the root of the tree.
+
+![Prolly Tree Value Update](../../.gitbook/assets/prolly-tree-update-value.png)
+
+Note, this is only true for fixed-size value updates. If you change the length of a value, then we do have to re-chunk. Changes to NULL values or strings are not fixed length updates.
+
+### Insert Keys
+
+We can insert keys at the beginning, middle, or end of the key space. It's helpful to visualize each type of insert.
+
+When we insert a key, we walk the tree by key to the leaf node where the key belongs. We then edit the chunk, calculate whether to split the chunk or not, and then recalculate the content address. Note, a key insert or a delete has a small probability of changing an existing chunk boundary. If the computed hash values for the keys, combined with the size of chunk at those keys, happens to choose a different chunk boundary, the chunk will be split at the new boundary. Finally, we walk up the tree recalculating each content address up to the root of the tree.
+
+In Dolt's Prolly Tree implementation, chunks are set to be on average 4 kilobytes. This means that you can imagine a probability of 1/4096 or 0.02% of triggering a chunk boundary shift when changing a single byte. In practice, it's quite a bit more complicated. As we'll discuss later, the size of the chunk is considered when computing a chunk boundary split. So, the larger the chunk, the greater the probability the chunk will be split on the addition of a new key. Conversely, for small chunks, the probability of a chunk split is very small. This means that every edit to a table in Dolt is a minimum of 4Kb multiplied by the depth of the tree with some small probability of writing multiple chunks.
+
+#### At the Beginning
+
+When you insert a key at the beginning of the key space, the tree is modified along the left edge.
+
+![Prolly Tree Beginning Insert](../../.gitbook/assets/prolly-tree-front-insert.png)
+
+#### At the end
+
+When you insert a key at the end of the key space, the tree is modified along the right edge.
+
+![Prolly Tree End Insert](../../.gitbook/assets/prolly-tree-back-insert.png)
+
+#### In the middle
+
+When you insert a key in the middle of the key space, the tree is modified along a spline.
+
+![Prolly Tree Middle Insert](../../.gitbook/assets/prolly-tree-middle-insert.png)
+
+### Delete a key
+
+When you delete a key, the tree is modified under the same rules as an insert.
+
+![Prolly Tree Delete Key](../../.gitbook/assets/prolly-tree-delete.png)
+
+# Properties
+
+## History Independence
+
+A key property of a Prolly tree that enables fast diff and structural sharing is history independence. No matter which order you insert, update, or delete values, the Prolly tree is the same. This is best seen through example.
+
+Consider a map with 4 integer keys, `(1, 2, 3, 4)` pointing at the same values. Before hand we know the combination of keys `(1, 2)` will trigger a chunk boundary. We can know this beforehand because chunk boundaries are based on the size of the chunk and its contents. No matter which order we insert, update, or delete, if we will end up with a tree with two leaf nodes `(1, 2)`, and `(3,4)`. This also happens to be true if the values at these keys are different. The tree will be the same but the the chunks will have different addresses.
+
+Let's say we insert the chunks in sequential order.
+
+![Prolly Tree History Independence](../../.gitbook/assets/prolly-tree-history-independence-1.png)
+
+Then, let's say we insert the keys in reverse order.
+
+![Prolly Tree History Independence](../../.gitbook/assets/prolly-tree-history-independence-2.png)
+
+As you can see, we end up with the same Prolly Tree no matter which order we insert the values. It's a fun exercise to try and come up with a sequence of inserts, updates, and deletes that result in a different tree that contains the same values. It's fun because you can't. The Prolly Tree algorithm always spits out the same tree.
+
+## Fast Diff
+
+Given history independence, the Prolly Tree difference calculation becomes quite simple. If the hash of the root chunk is the same the entire subtree is the same. Thus, one must just walk the tree down to the leaves, ignoring hashes that are equal. Hashes that are different represent the difference.
+
+Let's see how this algorithm works in practice. Recall the Prolly Tree where I updated the value in key 9. Let's compare it to our originally built Prolly Tree.
+
+![Prolly Tree Diff Step 1](../../.gitbook/assets/prolly-tree-diff-step-1.png)
+
+Start by comparing the root hashes. They are necessarily different.
+
+![Prolly Tree Diff Step 2](../../.gitbook/assets/prolly-tree-diff-step-2.png)
+
+Then walk both chunks identifying the subtree hashes that are different. In this case, a single hash is different.
+
+![Prolly Tree Diff Step 3](../../.gitbook/assets/prolly-tree-diff-step-3.png)
+
+Follow the pointer to the next layer and compare those chunks, finding the hashes that are different. If you are in a leaf node, produce the values that are different.
+
+![Prolly Tree Diff Step 4](../../.gitbook/assets/prolly-tree-diff-step-4.png)
+
+As you can see this algorithm scales with the size of the differences, not the size of the tree. This makes finding small differences in even large tress very fast.
## Structural Sharing
-To see how this structural sharing works in practice, let's examine a couple of scenarios for updating a prolly tree backing a Dolt table.
+Recall that Prolly trees are stored in a content addressed block store where the content address forms the lookup key for the block.
+
+![Content Addressed Block Store](../../.gitbook/assets/tim-prolly-tree-step-3.png)
+
+Thus, any blocks that share the same content address are only stored in the block store once. When they need to be retrieved they are retrieved via content address. This means that any blocks shared across versions will only be in the block store once.
+
+As noted earlier, in Dolt's Prolly Tree implementation, the block size is set to be on average 4 kilobytes. Thus, a single change will cause a change in the block store of 4 kilobytes times the size of the tree on average.
+
+# The Nitty Gritty Details
+
+Now that you have the general details, let's dive into some details highlighting some of the things we learned iterating on the Noms implementation of Prolly Trees and finally landing on Dolt's stable Prolly Tree implementation.
+
+## Controlling Chunk Size
+
+The original Noms implementation of Prolly Trees was susceptible to a chunk size problem. You would end up with many small chunks and a few large ones. The chunk size was a geometric distribution with an average size of 4 kilobytes.
+
+![Chunk Size Geometric](../../.gitbook/assets/chunk-distribution-static-pattern.png)
+
+This is the pattern one would expect from repeated rolls of a rolling hash function. Let's imagine you have a six-sided die. You walk along the road picking up pebbles. For each pebble, you put it in a bag and roll the die. If the die shows 6 you get a new bag. The average number of pebbles you have in each bag is 3.5 but you will mostly have bags with one pebble and a few bags with >10 pebbles. The pebbles in the Noms case was the key-value byte stream and the dice was the rolling hash function.
+
+Large chunks create a particular problem, especially on the read path, because you are more likely to be reading from large chunks. Within each chunk a binary search is performed to find the key you want. The larger the chunk the slower this is at log2(n) chunk size. So, you really want to keep a normally distributed chunk size.
+
+To fix this, Dolt's Prolly Tree implementation considers the chunk size when deciding whether to make a boundary. Given a target probability distribution function (PDF), Dolt uses its associated cumulative distribution function (CDF) to decide the probability of splitting a chunk of size x. Specifically, we use the formula `(CDF(end) - CDF(start)) / (1 - CDF(start))` to calculate the target probability. So, if the current chunk size is 2000 bytes the probability of triggering a boundary by appending a 64 byte key-value pair is `(CDF(2064) - CDF(2000) / 1 - CDF(2000)`. In Dolt, we want our chunk size normally distributed around 4 kilobytes and by using the above approach, we get that.
+
+![Chunk Size Normal](../../.gitbook/assets/chunk-distribution-dynamic-pattern.png)
+
+## Only Consider Keys
+
+Noms original Prolly Tree implementation considered keys and values when deciding when to make a chunk boundary. Dolt's implementation only considers keys.
+
+Dolt's Prolly Trees have the advantage of backing a SQL database. SQL databases define schema with fixed types. Fixed types have maximum sizes. Thus, any updates to values can be done in place because the values will be the same size. If we compute the rolling hash only keys, any update to values is guaranteed not to shift the chunk boundary. This is a desirable property that improves update performance.
+
+Moreover, Noms rolling hash function, [buzhash](https://github.com/silvasur/buzhash), performed poorly for byte streams with low entropy. Tables with ordered keys, specifically time series data where very little changes at each sample, were problematic. This would again lead to very large chunks as no chunk boundary was triggered because most of the byte stream considered by the rolling hash function was the same. By considering only keys, which again are necessarily unique, Dolt's hash function created chunk boundaries more normally.
+
+One subtlety of this change is that Dolt now chunks to an average number of key, values pairs rather than an average size of 4 kilobytes, but this difference disappears when used in concert with chunk size consideration in the probability of creating chunk boundaries.
+
+## Less Flexible Block Store
+
+Prolly Trees themselves are block store agnostic. So if you only care about the Prolly Tree data structure, you can skip this section.
+
+Noms block store encoded the types of the data in the store. Dolt's block store is built for a SQL database with fixed schema. Thus, type information is stored out of band of the block store and a [more specific, less flexible layout of data on disk is used](https://www.dolthub.com/blog/2022-05-20-new-format-alpha/). This new, type-less layout improves Dolt read and write performance.
+
+# Explaining Algorithmic Performance
+
+Recall this table:
+
+Operation | B-Trees | Prolly Trees
+--------- | ------- | ------------
+1 Random Read | logk(n) | logk(n)
+1 Random Write | logk(n) | (1+k/w)*logk(n)
+Ordered scan of one item with size z | z/k | z/k
+Calculate diff of size d | n | d
+Structural sharing | ❌ | ✅
+
+**n**: total leaf data in tree, **k**: average block size, **w**: window width
-* The absolute smallest overhead for any mutation to a prolly tree is, on average, a little larger than 4KB times the depth of the tree. Typically changing the value in a chunk won't change the chunk boundary, and you'll need to store complete new chunk values for the new leaf chunk and all the new internal chunks up to the root.
+We now understand what `n`, `k`, and `w` are in the context of B-Trees and Prolly Trees. As you can see, B-Trees and Prolly Trees offer similar read performance. Prolly Trees pay a slight performance penalty on writes due to the small probability of a chunk split. However, Prolly Trees can produce differences in time proportional to the size of the differences, rather than the size of the tree.
-The best case scenario looks like:
+# How Prolly Trees are used in Dolt
-![Single Value Edit](../../.gitbook/assets/single-value-edit.png)
+In Dolt, [all data in the database is stored in Prolly Trees](https://docs.dolthub.com/architecture/storage-engine#commit-graph).
-* Adding rows to a table whose primary keys are lexicographically larger than any of the existing rows in the table results in an append at the end of Prolly-tree for the table value. The last leaf node in the tree will necessarily change, and new chunks will be created for all of the new rows. Expected duplicate storage overhead is going to be the 4KB chunk at the end of the table's leaf nodes and the spline of the internal nodes of the map leading up to the root node. This kind of table might represent naturally append-only data or time-series data, and the storage overhead is very small. This looks like:
+For table data, a map of primary key to data columns is stored in a Prolly Tree. Similarly, secondary indexes are maps of index values to the primary key identifying each row. Schemas are stored as Prolly trees to make calculating the root of the database easy. Keyless tables are implemented as every column is a primary key with a count of the number of duplicate rows as the value.
-![Append Edit](../../.gitbook/assets/append-edit.png)
+# Prolly Trees In Practice
-* Adding rows to a table whose primary keys are lexicographically smaller than any of the existing rows is very similar to the case where they are all larger. It's expected to rewrite the first chunks in the table, and the probabilistic rolling hash will quickly resynchronize with the existing table data. It might look like:
+This all looks good on paper. How do Prolly Trees work in practice? On a standard suite of `sysbench` performance tests, Dolt is approximately [2X slower than MySQL](https://docs.dolthub.com/sql-reference/benchmarks/latency).
-![Prepend Edit](../../.gitbook/assets/prepend-edit.png)
+Upon profiling, we find most of the performance difference to be unrelated to Prolly Trees. The performance difference comes from:
-* Adding a run of data whose primary keys fall lexicographically within two existing rows is very similar to the prefix or postfix case. The run of data gets interpolated between the existing chunks of row data:
+1. Dolt is implemented in Golang. MySQL is implemented in C.
+2. MySQL's SQL analyzer is faster than Dolt's because it is more mature.
+3. MySQL does fewer transformations on data than Dolt to get it into the necessary wire format.
-![Middle Run Edit](../../.gitbook/assets/middle-run-edit.png)
\ No newline at end of file
+Dolt can [compute diffs in time proportional to the size of the differences](https://www.dolthub.com/blog/2022-06-03-dolt-diff-vs-sqlite-diff/). Dolt structurally shares [data across versions](https://www.dolthub.com/blog/2023-12-06-sizing-your-dolt-instance/#version-storage).
\ No newline at end of file
diff --git a/content/products/doltlab/installation.md b/content/products/doltlab/installation.md
index 66ddf24f6..c3595e71f 100644
--- a/content/products/doltlab/installation.md
+++ b/content/products/doltlab/installation.md
@@ -2,7 +2,7 @@
title: "Installation"
---
-The latest version of DoltLab is `v2.0.7` and to get started running your own DoltLab instance, you can follow the steps below. To see release notes for [DoltLab's releases](https://github.com/dolthub/doltlab-issues/releases) or to report and track DoltLab issues, visit DoltLab's [issues repository](https://github.com/dolthub/doltlab-issues).
+The latest version of DoltLab is `v2.0.8` and to get started running your own DoltLab instance, you can follow the steps below. To see release notes for [DoltLab's releases](https://github.com/dolthub/doltlab-issues/releases) or to report and track DoltLab issues, visit DoltLab's [issues repository](https://github.com/dolthub/doltlab-issues).
Please note, that to upgrading to a newer version of DoltLab will require you to kill the older version of DoltLab and install the newer one, which may result in data loss.
@@ -39,7 +39,7 @@ If your host is running Ubuntu 18.04/20.04, the quickest way to install these de
To use them:
```bash
-export DOLTLAB_VERSION=v2.0.7
+export DOLTLAB_VERSION=v2.0.8
chmod +x ubuntu-bootstrap.sh
sudo ./ubuntu-bootstrap.sh with-sudo "$DOLTLAB_VERSION"
cd doltlab
@@ -47,7 +47,7 @@ sudo newgrp docker # login as root to run docker without sudo
```
```bash
-export DOLTLAB_VERSION=v2.0.7
+export DOLTLAB_VERSION=v2.0.8
chmod +x centos-bootstrap.sh
sudo ./centos-bootstrap.sh with-sudo "$DOLTLAB_VERSION"
cd doltlab
@@ -81,7 +81,7 @@ cd doltlab
To install a specific version, run:
```bash
-export DOLTLAB_VERSION=v2.0.7
+export DOLTLAB_VERSION=v2.0.8
curl -LO https://doltlab-releases.s3.amazonaws.com/linux/amd64/doltlab-${DOLTLAB_VERSION}.zip
unzip doltlab-${DOLTLAB_VERSION}.zip -d doltlab
cd doltlab
diff --git a/content/reference/sql/benchmarks/correctness.md b/content/reference/sql/benchmarks/correctness.md
index b4e3a54d4..678c0bb53 100644
--- a/content/reference/sql/benchmarks/correctness.md
+++ b/content/reference/sql/benchmarks/correctness.md
@@ -53,18 +53,18 @@ AND col3 IN (3,9,0))))) OR col4 <= 4.25 OR ((col3 = 5))) OR (((col0 >
0)) AND col0 > 6 AND (col4 >= 6.56)))
```
-Here are Dolt's sqllogictest results for version `1.34.0`. Tests that
+Here are Dolt's sqllogictest results for version `1.34.2`. Tests that
did not run could not complete due to a timeout earlier in the run.
| Results | Count |
|---------|---------|
-| not ok | 26 |
-| ok | 5937431 |
+| not ok | 9 |
+| ok | 5937448 |
| Total Tests | 5937457 |
|-------------|---------|
-| Correctness Percentage | 99.999562 |
+| Correctness Percentage | 99.999848 |
|------------------------|-----------|
diff --git a/content/reference/sql/benchmarks/latency.md b/content/reference/sql/benchmarks/latency.md
index 16981e637..34cd144ff 100644
--- a/content/reference/sql/benchmarks/latency.md
+++ b/content/reference/sql/benchmarks/latency.md
@@ -28,36 +28,41 @@ attempt to run as many queries as possible in a fixed 2 minute time
window. The `Dolt` and `MySQL` columns show the median latency in
milliseconds (ms) of each query during that 2 minute time window.
-The Dolt version is `1.34.0`.
+The Dolt version is `1.34.2`.
-| Read Tests | MySQL | Dolt | Multiple |
-|-------------------------|-------|--------|----------|
-| covering\_index\_scan | 2.11 | 3.02 | 1.4 |
-| groupby\_scan | 13.7 | 18.28 | 1.3 |
-| index\_join | 1.34 | 5.28 | 3.9 |
-| index\_join\_scan | 1.25 | 2.22 | 1.8 |
-| index\_scan | 33.72 | 62.19 | 1.8 |
-| oltp\_point\_select | 0.17 | 0.47 | 2.8 |
-| oltp\_read\_only | 3.36 | 8.13 | 2.4 |
-| select\_random\_points | 0.32 | 0.77 | 2.4 |
-| select\_random\_ranges | 0.39 | 0.9 | 2.3 |
-| table\_scan | 34.33 | 62.19 | 1.8 |
-| types\_table\_scan | 74.46 | 170.48 | 2.3 |
-| reads\_mean\_multiplier | | | 2.2 |
+| Read Tests | MySQL | Dolt | Multiple |
+|-------------------------|-------|-------|----------|
+| covering\_index\_scan | 2.11 | 2.81 | 1.3 |
+| groupby\_scan | 13.7 | 17.01 | 1.2 |
+| index\_join | 1.32 | 4.91 | 3.7 |
+| index\_join\_scan | 1.25 | 2.14 | 1.7 |
+| index\_scan | 33.72 | 57.87 | 1.7 |
+| oltp\_point\_select | 0.17 | 0.46 | 2.7 |
+| oltp\_read\_only | 3.36 | 8.13 | 2.4 |
+| select\_random\_points | 0.33 | 0.77 | 2.3 |
+| select\_random\_ranges | 0.38 | 0.87 | 2.3 |
+| table\_scan | 34.33 | 57.87 | 1.7 |
+| types\_table\_scan | 74.46 | 155.8 | 2.1 |
+| reads\_mean\_multiplier | | | 2.1 |
| Write Tests | MySQL | Dolt | Multiple |
|--------------------------|-------|-------|----------|
-| oltp\_delete\_insert | 5.88 | 5.99 | 1.0 |
-| oltp\_insert | 2.81 | 2.97 | 1.1 |
-| oltp\_read\_write | 7.43 | 15.55 | 2.1 |
-| oltp\_update\_index | 2.86 | 3.13 | 1.1 |
-| oltp\_update\_non\_index | 2.97 | 3.07 | 1.0 |
-| oltp\_write\_only | 4.03 | 7.43 | 1.8 |
-| types\_delete\_insert | 5.47 | 6.67 | 1.2 |
-| writes\_mean\_multiplier | | | 1.3 |
-
-| Overall Mean Multiple | 1.9 |
-|-----------------------|-----|
+| oltp\_delete\_insert | 5.67 | 5.99 | 1.1 |
+| oltp\_insert | 2.76 | 3.02 | 1.1 |
+| oltp\_read\_write | 7.3 | 15.27 | 2.1 |
+| oltp\_update\_index | 2.81 | 3.07 | 1.1 |
+| oltp\_update\_non\_index | 2.86 | 3.02 | 1.1 |
+| oltp\_write\_only | 3.96 | 7.43 | 1.9 |
+| types\_delete\_insert | 5.57 | 6.67 | 1.2 |
+| writes\_mean\_multiplier | | | 1.4 |
+
+| TPC-C TPS Tests | MySQL | Dolt | Multiple |
+|-----------------------|--------|------|----------|
+| tpcc-scale-factor-1 | 110.64 | 15.5 | 6.5 |
+| tpcc\_tps\_multiplier | | | 6.5 |
+
+| Overall Mean Multiple | 3.33 |
+|-----------------------|------|
\ No newline at end of file
diff --git a/content/reference/sql/server/troubleshooting.md b/content/reference/sql/server/troubleshooting.md
index ab37d6d25..e77ac23a0 100644
--- a/content/reference/sql/server/troubleshooting.md
+++ b/content/reference/sql/server/troubleshooting.md
@@ -90,36 +90,3 @@ depending on the size of the database, the size of the updates, replication sett
factors.
Improving maximum write concurrency is an ongoing project.
-
-## Session state leaking due to connection pooling
-
-Many SQL database client libraries implement some kind of connection pool, where connections to the
-database are created ahead of time and then handed out to different execution threads when needed to
-execute queries. This is typically done to reduce latency on database operations. Usually these pools
-reuse connections after they are returned to the pool.
-
-Using connection pooling with Dolt when combined with `dolt_checkout()` can be problematic, because
-a session that gets returned to the pool and re-used may behave differently (modify a different
-branch head) than a fresh connection. Consider this sequence:
-
-```sql
-call dolt_checkout('feature-branch');
-...
-call dolt_commit();
-```
-
-When this connection is returned to the pool, its session has `feature-branch` checked out. If it's
-reused by another execution thread that expects a different branch, you have a bug.
-
-To mitigate potential bugs with the checked out branch in session state, we recommend that you
-either:
-
-* Disable connection pooling if feasible
-* Configure your connection pool to not re-use connections that were returned to it
-* Use a different connection string for each branch (this may mean using multiple connection pools)
-* Augment your application logic to always begin a new unit of work with `call dolt_checkout(...)`
- to ensure the expected branch is checked out for the session
-
-[Implementing the `COM_RESET_CONNECTION` protocol](https://github.com/dolthub/dolt/issues/3921)
-would solve the problem of leaking session state between different execution threads, but support
-for this functionality in popular libraries is spotty. For now, use the above guidance.
diff --git a/content/reference/sql/sql-support/supported-statements.md b/content/reference/sql/sql-support/supported-statements.md
index 412ea5e2c..efa925cd4 100644
--- a/content/reference/sql/sql-support/supported-statements.md
+++ b/content/reference/sql/sql-support/supported-statements.md
@@ -28,7 +28,7 @@ title: Supported Statements
| `UPDATE` | ✅ | No support for referring to more than one table in a single `UPDATE` statement. |
| `VALUES` | ✅ | |
| `WITH` | ✅ | |
-| `SELECT INTO` | ✅ | Selecting into a file is not supported. |
+| `SELECT INTO` | ✅ | Charset/Collation specification not supported. |
## Data definition statements
diff --git a/content/reference/sql/version-control/dolt-sql-procedures.md b/content/reference/sql/version-control/dolt-sql-procedures.md
index 121b3753d..d2c931ce2 100644
--- a/content/reference/sql/version-control/dolt-sql-procedures.md
+++ b/content/reference/sql/version-control/dolt-sql-procedures.md
@@ -545,6 +545,8 @@ CALL DOLT_CLONE('dolthub/us-jails', 'myCustomDbName');
`-b`, `--branch`: The branch to be cloned. If not specified all branches will be cloned.
+`--depth`: Clone a single branch and limit history to the given commit depth.
+
### Output Schema
```text
diff --git a/update-perf.sh b/update-perf.sh
index 39636ed6d..96dafee84 100755
--- a/update-perf.sh
+++ b/update-perf.sh
@@ -14,19 +14,18 @@ os_type="darwin"
sql_reference_dir="content/reference/sql/benchmarks"
-new_table="updated.md"
-
start_template=''
end_template=''
-if [ "$#" -ne 3 ]; then
- echo "Must supply version and type, eg update-perf.sh 'v0.39.0' '__LD_1__|__DOLT__' 'latency|correctness'"
+if [ "$#" -ne 4 ]; then
+ echo "usage: update-perf.sh "
exit 1;
fi
version="$1"
storage_format="$2"
type="$3"
+new_table="$4"
if [[ "$OSTYPE" == "linux-gnu"* ]]; then
os_type="linux"