salsa/tests/incremental/memoized_dep_inputs.rs

72 lines
2 KiB
Rust

use crate::implementation::{TestContext, TestContextImpl};
use salsa::Database;
salsa::query_group! {
pub(crate) trait MemoizedDepInputsContext: TestContext {
fn dep_memoized2() -> usize {
type Memoized2;
}
fn dep_memoized1() -> usize {
type Memoized1;
}
fn dep_derived1() -> usize {
type Derived1;
storage dependencies;
}
fn dep_input1() -> usize {
type Input1;
storage input;
}
fn dep_input2() -> usize {
type Input2;
storage input;
}
}
}
fn dep_memoized2(db: &impl MemoizedDepInputsContext) -> usize {
db.log().add("Memoized2 invoked");
db.dep_memoized1()
}
fn dep_memoized1(db: &impl MemoizedDepInputsContext) -> usize {
db.log().add("Memoized1 invoked");
db.dep_derived1() * 2
}
fn dep_derived1(db: &impl MemoizedDepInputsContext) -> usize {
db.log().add("Derived1 invoked");
db.dep_input1() / 2
}
#[test]
fn revalidate() {
let db = &mut TestContextImpl::default();
db.query_mut(Input1).set((), 0);
// Initial run starts from Memoized2:
let v = db.dep_memoized2();
assert_eq!(v, 0);
db.assert_log(&["Memoized2 invoked", "Memoized1 invoked", "Derived1 invoked"]);
// After that, we first try to validate Memoized1 but wind up
// running Memoized2. Note that we don't try to validate
// Derived1, so it is invoked by Memoized1.
db.query_mut(Input1).set((), 44);
let v = db.dep_memoized2();
assert_eq!(v, 44);
db.assert_log(&["Memoized1 invoked", "Derived1 invoked", "Memoized2 invoked"]);
// Here validation of Memoized1 succeeds so Memoized2 never runs.
db.query_mut(Input1).set((), 45);
let v = db.dep_memoized2();
assert_eq!(v, 44);
db.assert_log(&["Memoized1 invoked", "Derived1 invoked"]);
// Here, a change to input2 doesn't affect us, so nothing runs.
db.query_mut(Input2).set((), 45);
let v = db.dep_memoized2();
assert_eq!(v, 44);
db.assert_log(&[]);
}