Skip to content

Commit

Permalink
refactor: Implement try_take_leased_lock on MemoryStore.
Browse files Browse the repository at this point in the history
  • Loading branch information
Hywan committed Nov 5, 2024
1 parent a8f244a commit 05f275d
Showing 1 changed file with 49 additions and 3 deletions.
52 changes: 49 additions & 3 deletions crates/matrix-sdk-base/src/event_cache_store/memory_store.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,12 @@
// See the License for the specific language governing permissions and
// limitations under the License.

use std::{num::NonZeroUsize, sync::RwLock as StdRwLock};
use std::{
collections::{hash_map::Entry, HashMap},
num::NonZeroUsize,
sync::RwLock as StdRwLock,
time::{Duration, Instant},
};

use async_trait::async_trait;
use matrix_sdk_common::ring_buffer::RingBuffer;
Expand All @@ -28,14 +33,18 @@ use crate::media::{MediaRequest, UniqueKey as _};
#[derive(Debug)]
pub struct MemoryStore {
media: StdRwLock<RingBuffer<(OwnedMxcUri, String /* unique key */, Vec<u8>)>>,
leases: StdRwLock<HashMap<String, (String, Instant)>>,
}

// SAFETY: `new_unchecked` is safe because 20 is not zero.
const NUMBER_OF_MEDIAS: NonZeroUsize = unsafe { NonZeroUsize::new_unchecked(20) };

impl Default for MemoryStore {
fn default() -> Self {
Self { media: StdRwLock::new(RingBuffer::new(NUMBER_OF_MEDIAS)) }
Self {
media: StdRwLock::new(RingBuffer::new(NUMBER_OF_MEDIAS)),
leases: Default::default(),
}
}
}

Expand All @@ -57,7 +66,44 @@ impl EventCacheStore for MemoryStore {
key: &str,
holder: &str,
) -> Result<bool, Self::Error> {
todo!()
let now = Instant::now();
let expiration = now + Duration::from_millis(lease_duration_ms.into());

match self.leases.write().unwrap().entry(key.to_owned()) {
// There is an existing holder.
Entry::Occupied(mut entry) => {
let (current_holder, current_expiration) = entry.get_mut();

if current_holder == holder {
// We had the lease before, extend it.
*current_expiration = expiration;

Ok(true)
} else {
// We didn't have it.
if *current_expiration < now {
// Steal it!
*current_holder = holder.to_owned();
*current_expiration = expiration;

Ok(true)
} else {
// We tried our best.
Ok(false)
}
}
}

// There is no holder, easy.
Entry::Vacant(entry) => {
entry.insert((
holder.to_owned(),
Instant::now() + Duration::from_millis(lease_duration_ms.into()),
));

Ok(true)
}
}
}

async fn add_media_content(&self, request: &MediaRequest, data: Vec<u8>) -> Result<()> {
Expand Down

0 comments on commit 05f275d

Please sign in to comment.