Remove references to old selection example

This commit is contained in:
Matthijs Brobbel 2024-06-18 10:39:26 +02:00
parent a0d7b0ee55
commit d0e9b79e6a
No known key found for this signature in database
GPG key ID: 551189F8515034D3
3 changed files with 2 additions and 87 deletions

View file

@ -19,7 +19,6 @@
- [Durability](./reference/durability.md)
- [Algorithm](./reference/algorithm.md)
- [Common patterns](./common_patterns.md)
- [Selection](./common_patterns/selection.md)
- [On-demand (Lazy) inputs](./common_patterns/on_demand_inputs.md)
- [Tuning](./tuning.md)
- [Cycle handling](./cycles.md)

View file

@ -1,78 +0,0 @@
# Selection
The "selection" (or "firewall") pattern is when you have a query Qsel that reads from some
other Qbase and extracts some small bit of information from Qbase that it returns.
In particular, Qsel does not combine values from other queries. In some sense,
then, Qsel is redundant -- you could have just extracted the information
the information from Qbase yourself, and done without the salsa machinery. But
Qsel serves a role in that it limits the amount of re-execution that is required
when Qbase changes.
## Example: the base query
For example, imagine that you have a query `parse` that parses the input text of a request
and returns a `ParsedResult`, which contains a header and a body:
```rust,ignore
{{#include ../../../examples/selection/main.rs:request}}
```
## Example: a selecting query
And now you have a number of derived queries that only look at the header.
For example, one might extract the "content-type' header:
```rust,ignore
{{#include ../../../examples/selection/util1.rs:util1}}
```
## Why prefer a selecting query?
This `content_type` query is an instance of the *selection* pattern. It only
"selects" a small bit of information from the `ParsedResult`. You might not have
made it a query at all, but instead made it a method on `ParsedResult`.
But using a query for `content_type` has an advantage: now if there are downstream
queries that only depend on the `content_type` (or perhaps on other headers extracted
via a similar pattern), those queries will not have to be re-executed when the request
changes *unless* the content-type header changes. Consider the dependency graph:
```text
request_text --> parse --> content_type --> (other queries)
```
When the `request_text` changes, we are always going to have to re-execute `parse`.
If that produces a new parsed result, we are *also* going to re-execute `content_type`.
But if the result of `content_type` has not changed, then we will *not* re-execute
the other queries.
## More levels of selection
In fact, in our example we might consider introducing another level of selection.
Instead of having `content_type` directly access the results of `parse`, it might be better
to insert a selecting query that just extracts the header:
```rust,ignore
{{#include ../../../examples/selection/util2.rs:util2}}
```
This will result in a dependency graph like so:
```text
request_text --> parse --> header --> content_type --> (other queries)
```
The advantage of this is that changes that only effect the "body" or
only consume small parts of the request will
not require us to re-execute `content_type` at all. This would be particularly
valuable if there are a lot of dependent headers.
## A note on cloning and efficiency
In this example, we used common Rust types like `Vec` and `String`,
and we cloned them quite frequently. This will work just fine in Salsa,
but it may not be the most efficient choice. This is because each clone
is going to produce a deep copy of the result. As a simple fix, you
might convert your data structures to use `Arc` (e.g., `Arc<Vec<ParsedHeader>>`),
which makes cloning cheap.

View file

@ -25,15 +25,10 @@ Interning is especially useful for queries that involve nested,
tree-like data structures.
See:
- The [`compiler` example](https://github.com/salsa-rs/salsa/blob/master/examples/compiler/main.rs),
which uses interning.
## Granularity of Incrementality
See:
- [common patterns: selection](./common_patterns/selection.md) and
- The [`selection` example](https://github.com/salsa-rs/salsa/blob/master/examples/selection/main.rs)
## Cancellation
Queries that are no longer needed due to concurrent writes or changes in dependencies are cancelled
@ -45,4 +40,3 @@ salsa won't be able to cancel it automatically. You may wish to check for cancel
by invoking `db.unwind_if_cancelled()`.
For more details on cancellation, see the tests for cancellation behavior in the Salsa repo.