use std::collections::BTreeMap;
use std::fmt::{Debug, Formatter};
use std::hash::Hash;
use std::sync::Arc;
use std::time::Duration;
use differential_dataflow::consolidation::{consolidate, consolidate_updates};
use mz_ore::collections::{AssociativeExt, HashSet};
use mz_ore::soft_panic_or_log;
use mz_persist_client::critical::SinceHandle;
use mz_persist_client::error::UpperMismatch;
use mz_persist_client::read::{ListenEvent, Subscribe};
use mz_persist_client::write::WriteHandle;
use mz_persist_client::{Diagnostics, PersistClient};
use mz_persist_types::{Codec, ShardId};
use timely::progress::Antichain;
use tracing::debug;
pub trait DurableCacheCodec: Debug + Eq {
type Key: Ord + Hash + Clone + Debug;
type Val: Eq + Debug;
type KeyCodec: Codec + Ord + Debug + Clone;
type ValCodec: Codec + Ord + Debug + Clone;
fn schemas() -> (
<Self::KeyCodec as Codec>::Schema,
<Self::ValCodec as Codec>::Schema,
);
fn encode(key: &Self::Key, val: &Self::Val) -> (Self::KeyCodec, Self::ValCodec);
fn decode(key: &Self::KeyCodec, val: &Self::ValCodec) -> (Self::Key, Self::Val);
}
#[derive(Debug)]
pub enum Error {
WriteConflict(UpperMismatch<u64>),
}
impl std::fmt::Display for Error {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self {
Error::WriteConflict(err) => write!(f, "{err}"),
}
}
}
#[derive(Debug, PartialEq, Eq)]
struct LocalVal<C: DurableCacheCodec> {
encoded_key: C::KeyCodec,
decoded_val: C::Val,
encoded_val: C::ValCodec,
}
#[derive(Debug)]
pub struct DurableCache<C: DurableCacheCodec> {
since_handle: SinceHandle<C::KeyCodec, C::ValCodec, u64, i64, i64>,
write: WriteHandle<C::KeyCodec, C::ValCodec, u64, i64>,
subscribe: Subscribe<C::KeyCodec, C::ValCodec, u64, i64>,
local: BTreeMap<C::Key, LocalVal<C>>,
local_progress: u64,
}
const USE_CRITICAL_SINCE: bool = true;
impl<C: DurableCacheCodec> DurableCache<C> {
pub async fn new(persist: &PersistClient, shard_id: ShardId, purpose: &str) -> Self {
let diagnostics = Diagnostics {
shard_name: format!("{purpose}_cache"),
handle_purpose: format!("durable persist cache: {purpose}"),
};
let since_handle = persist
.open_critical_since(
shard_id,
PersistClient::CONTROLLER_CRITICAL_SINCE,
diagnostics.clone(),
)
.await
.expect("invalid usage");
let (key_schema, val_schema) = C::schemas();
let (mut write, read) = persist
.open(
shard_id,
Arc::new(key_schema),
Arc::new(val_schema),
diagnostics,
USE_CRITICAL_SINCE,
)
.await
.expect("shard codecs should not change");
let res = write
.compare_and_append_batch(&mut [], Antichain::from_elem(0), Antichain::from_elem(1))
.await
.expect("usage was valid");
match res {
Ok(()) => {}
Err(UpperMismatch { .. }) => {}
}
let as_of = read.since().clone();
let subscribe = read
.subscribe(as_of)
.await
.expect("capability should be held at this since");
let mut ret = DurableCache {
since_handle,
write,
subscribe,
local: BTreeMap::new(),
local_progress: 0,
};
ret.sync_to(ret.write.upper().as_option().copied()).await;
ret
}
async fn sync_to(&mut self, progress: Option<u64>) -> u64 {
let progress = progress.expect("cache shard should not be closed");
while self.local_progress < progress {
let events = self.subscribe.fetch_next().await;
for event in events {
match event {
ListenEvent::Updates(mut x) => {
consolidate_updates(&mut x);
x.sort_by(|(_, ts1, d1), (_, ts2, d2)| ts1.cmp(ts2).then(d1.cmp(d2)));
for ((k, v), t, d) in x {
let encoded_key = k.unwrap();
let encoded_val = v.unwrap();
let (decoded_key, decoded_val) = C::decode(&encoded_key, &encoded_val);
let val = LocalVal {
encoded_key,
decoded_val,
encoded_val,
};
if d == 1 {
self.local
.expect_insert(decoded_key, val, "duplicate cache entry");
} else if d == -1 {
let prev = self
.local
.expect_remove(&decoded_key, "entry does not exist");
assert_eq!(val, prev, "removed val does not match expected val");
} else {
panic!(
"unexpected diff: (({:?}, {:?}), {}, {})",
decoded_key, val.decoded_val, t, d
);
}
}
}
ListenEvent::Progress(x) => {
self.local_progress =
x.into_option().expect("cache shard should not be closed")
}
}
}
}
progress
}
pub fn get_local(&self, key: &C::Key) -> Option<&C::Val> {
self.local.get(key).map(|val| &val.decoded_val)
}
pub async fn get(&mut self, key: &C::Key, val_fn: impl FnOnce() -> C::Val) -> &C::Val {
if self.local.contains_key(key) {
return self.get_local(key).expect("checked above");
}
self.sync_to(self.write.shared_upper().into_option()).await;
if self.local.contains_key(key) {
return self.get_local(key).expect("checked above");
}
let val = val_fn();
let mut expected_upper = self.local_progress;
let update = (C::encode(key, &val), 1);
loop {
let ret = self
.compare_and_append([update.clone()], expected_upper)
.await;
match ret {
Ok(new_upper) => {
self.sync_to(Some(new_upper)).await;
return self.get_local(key).expect("just inserted");
}
Err(err) => {
expected_upper = self.sync_to(err.current.into_option()).await;
if self.local.contains_key(key) {
return self.get_local(key).expect("checked above");
}
continue;
}
}
}
}
pub fn entries_local(&self) -> impl Iterator<Item = (&C::Key, &C::Val)> {
self.local.iter().map(|(key, val)| (key, &val.decoded_val))
}
pub async fn set(&mut self, key: &C::Key, value: Option<&C::Val>) {
while let Err(err) = self.try_set(key, value).await {
debug!("failed to set entry: {err} ... retrying");
}
}
pub async fn set_many(&mut self, entries: &[(&C::Key, Option<&C::Val>)]) {
while let Err(err) = self.try_set_many(entries).await {
debug!("failed to set entries: {err} ... retrying");
}
}
pub async fn try_set(&mut self, key: &C::Key, value: Option<&C::Val>) -> Result<(), Error> {
self.try_set_many(&[(key, value)]).await
}
pub async fn try_set_many(
&mut self,
entries: &[(&C::Key, Option<&C::Val>)],
) -> Result<(), Error> {
let expected_upper = self.local_progress;
let mut updates = Vec::new();
let mut seen_keys = HashSet::new();
for (key, val) in entries {
if seen_keys.insert(key) {
if let Some(prev) = self.local.get(key) {
updates.push(((prev.encoded_key.clone(), prev.encoded_val.clone()), -1));
}
if let Some(val) = val {
updates.push((C::encode(key, val), 1));
}
}
}
consolidate(&mut updates);
let ret = self.compare_and_append(updates, expected_upper).await;
match ret {
Ok(new_upper) => {
self.sync_to(Some(new_upper)).await;
Ok(())
}
Err(err) => {
self.sync_to(err.current.clone().into_option()).await;
Err(Error::WriteConflict(err))
}
}
}
async fn compare_and_append<I>(
&mut self,
updates: I,
write_ts: u64,
) -> Result<u64, UpperMismatch<u64>>
where
I: IntoIterator<Item = ((C::KeyCodec, C::ValCodec), i64)>,
{
let expected_upper = write_ts;
let new_upper = expected_upper + 1;
let updates = updates.into_iter().map(|((k, v), d)| ((k, v), write_ts, d));
self.write
.compare_and_append(
updates,
Antichain::from_elem(expected_upper),
Antichain::from_elem(new_upper),
)
.await
.expect("usage should be valid")?;
let downgrade_to = Antichain::from_elem(write_ts);
let opaque = *self.since_handle.opaque();
let ret = self
.since_handle
.compare_and_downgrade_since(&opaque, (&opaque, &downgrade_to))
.await;
if let Err(e) = ret {
soft_panic_or_log!("found opaque value {e}, but expected {opaque}");
}
Ok(new_upper)
}
pub async fn dangerous_compact_shard(
&self,
fuel: impl Fn() -> usize,
wait: impl Fn() -> Duration,
) {
mz_persist_client::cli::admin::dangerous_force_compaction_and_break_pushdown(
&self.write,
fuel,
wait,
)
.await
}
}
#[cfg(test)]
mod tests {
use mz_ore::assert_none;
use mz_persist_client::cache::PersistClientCache;
use mz_persist_types::codec_impls::StringSchema;
use mz_persist_types::PersistLocation;
use super::*;
#[derive(Debug, PartialEq, Eq)]
struct TestCodec;
impl DurableCacheCodec for TestCodec {
type Key = String;
type Val = String;
type KeyCodec = String;
type ValCodec = String;
fn schemas() -> (
<Self::KeyCodec as Codec>::Schema,
<Self::ValCodec as Codec>::Schema,
) {
(StringSchema, StringSchema)
}
fn encode(key: &Self::Key, val: &Self::Val) -> (Self::KeyCodec, Self::ValCodec) {
(key.clone(), val.clone())
}
fn decode(key: &Self::KeyCodec, val: &Self::ValCodec) -> (Self::Key, Self::Val) {
(key.clone(), val.clone())
}
}
#[mz_ore::test(tokio::test)]
#[cfg_attr(miri, ignore)]
async fn durable_cache() {
let persist = PersistClientCache::new_no_metrics();
let persist = persist
.open(PersistLocation::new_in_mem())
.await
.expect("location should be valid");
let shard_id = ShardId::new();
let mut cache0 = DurableCache::<TestCodec>::new(&persist, shard_id, "test1").await;
assert_none!(cache0.get_local(&"foo".into()));
assert_eq!(cache0.get(&"foo".into(), || "bar".into()).await, "bar");
assert_eq!(
cache0.entries_local().collect::<Vec<_>>(),
vec![(&"foo".into(), &"bar".into())]
);
cache0.set(&"k1".into(), Some(&"v1".into())).await;
cache0.set(&"k2".into(), Some(&"v2".into())).await;
assert_eq!(cache0.get_local(&"k1".into()), Some(&"v1".into()));
assert_eq!(cache0.get(&"k1".into(), || "v10".into()).await, &"v1");
assert_eq!(cache0.get_local(&"k2".into()), Some(&"v2".into()));
assert_eq!(cache0.get(&"k2".into(), || "v20".into()).await, &"v2");
assert_eq!(
cache0.entries_local().collect::<Vec<_>>(),
vec![
(&"foo".into(), &"bar".into()),
(&"k1".into(), &"v1".into()),
(&"k2".into(), &"v2".into())
]
);
cache0.set(&"k1".into(), None).await;
assert_none!(cache0.get_local(&"k1".into()));
assert_eq!(
cache0.entries_local().collect::<Vec<_>>(),
vec![(&"foo".into(), &"bar".into()), (&"k2".into(), &"v2".into())]
);
cache0
.set_many(&[
(&"k1".into(), Some(&"v10".into())),
(&"k2".into(), None),
(&"k3".into(), None),
])
.await;
assert_eq!(cache0.get_local(&"k1".into()), Some(&"v10".into()));
assert_none!(cache0.get_local(&"k2".into()));
assert_none!(cache0.get_local(&"k3".into()));
assert_eq!(
cache0.entries_local().collect::<Vec<_>>(),
vec![
(&"foo".into(), &"bar".into()),
(&"k1".into(), &"v10".into()),
]
);
cache0
.set_many(&[
(&"k4".into(), Some(&"v40".into())),
(&"k4".into(), Some(&"v41".into())),
(&"k4".into(), Some(&"v42".into())),
(&"k5".into(), Some(&"v50".into())),
(&"k5".into(), Some(&"v51".into())),
(&"k5".into(), Some(&"v52".into())),
])
.await;
assert_eq!(cache0.get_local(&"k4".into()), Some(&"v40".into()));
assert_eq!(cache0.get_local(&"k5".into()), Some(&"v50".into()));
assert_eq!(
cache0.entries_local().collect::<Vec<_>>(),
vec![
(&"foo".into(), &"bar".into()),
(&"k1".into(), &"v10".into()),
(&"k4".into(), &"v40".into()),
(&"k5".into(), &"v50".into()),
]
);
let mut cache1 = DurableCache::<TestCodec>::new(&persist, shard_id, "test2").await;
assert_eq!(cache1.get(&"foo".into(), || panic!("boom")).await, "bar");
assert_eq!(cache1.get(&"k1".into(), || panic!("boom")).await, &"v10");
assert_none!(cache1.get_local(&"k2".into()));
assert_none!(cache1.get_local(&"k3".into()));
assert_eq!(
cache1.entries_local().collect::<Vec<_>>(),
vec![
(&"foo".into(), &"bar".into()),
(&"k1".into(), &"v10".into()),
(&"k4".into(), &"v40".into()),
(&"k5".into(), &"v50".into()),
]
);
let fuel = || 131_072;
let wait = || Duration::from_millis(0);
cache1.dangerous_compact_shard(fuel, wait).await
}
}