Skip to content

Commit

Permalink
Add request count tests for FS operations with metadata caching enabl…
Browse files Browse the repository at this point in the history
…ed (#567)

* Add counters to MockClient for testing request counts

Signed-off-by: Daniel Carl Jones <[email protected]>

* Add tests verifying request counts after readdir,open,lookup,unlink with caching enabled

Signed-off-by: Daniel Carl Jones <[email protected]>

* Appease rustc warnings

Signed-off-by: Daniel Carl Jones <[email protected]>

* Appease clippy warnings

Signed-off-by: Daniel Carl Jones <[email protected]>

* Update OperationCounter::count rustdoc

Signed-off-by: Daniel Carl Jones <[email protected]>

* Add test verifying request counts on open when cache is off

Signed-off-by: Daniel Carl Jones <[email protected]>

---------

Signed-off-by: Daniel Carl Jones <[email protected]>
  • Loading branch information
dannycjones authored Oct 25, 2023
1 parent eac02e6 commit fa0d516
Show file tree
Hide file tree
Showing 2 changed files with 340 additions and 3 deletions.
86 changes: 85 additions & 1 deletion mountpoint-s3-client/src/mock_client.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
//! A mock implementation of an object client for use in tests.
use std::borrow::Cow;
use std::collections::{BTreeMap, BTreeSet};
use std::collections::{BTreeMap, BTreeSet, HashMap};
use std::ops::Range;
use std::pin::Pin;
use std::sync::{Arc, RwLock};
Expand Down Expand Up @@ -60,6 +60,7 @@ pub struct MockClient {
config: MockClientConfig,
objects: Arc<RwLock<BTreeMap<String, MockObject>>>,
in_progress_uploads: Arc<RwLock<BTreeSet<String>>>,
operation_counts: Arc<RwLock<HashMap<Operation, u64>>>,
}

fn add_object(objects: &Arc<RwLock<BTreeMap<String, MockObject>>>, key: &str, value: MockObject) {
Expand All @@ -73,6 +74,7 @@ impl MockClient {
config,
objects: Default::default(),
in_progress_uploads: Default::default(),
operation_counts: Default::default(),
}
}

Expand Down Expand Up @@ -134,6 +136,55 @@ impl MockClient {
Err(MockClientError("object not found".into()))
}
}

/// Create a new counter for the given operation, starting at 0.
pub fn new_counter(&self, operation: Operation) -> OperationCounter<'_> {
let op_counts = self.operation_counts.read().unwrap();
let initial_count = op_counts.get(&operation).copied().unwrap_or_default();

OperationCounter {
client: self,
initial_count,
operation,
}
}

/// Track number of operations for verifying API calls made by the client in testing.
fn inc_op_count(&self, operation: Operation) {
let mut op_counts = self.operation_counts.write().unwrap();
op_counts.entry(operation).and_modify(|count| *count += 1).or_insert(1);
}
}

/// Operations for use in operation counters.
#[derive(Debug, Eq, Hash, PartialEq)]
pub enum Operation {
DeleteObject,
HeadObject,
GetObject,
GetObjectAttributes,
ListObjectsV2,
PutObject,
}

/// Counter for a specific client [Operation].
///
/// Obtainable via `new_counter(&Operation)` method on [MockClient]
/// Its lifetime is bounded by the client which created it.
pub struct OperationCounter<'a> {
client: &'a MockClient,
initial_count: u64,
operation: Operation,
}

impl<'a> OperationCounter<'a> {
/// Return number of requests since the counter was created.
/// The counter is **not** reset when read.
pub fn count(&self) -> u64 {
let op_counts = self.client.operation_counts.read().unwrap();
let total_count = op_counts.get(&self.operation).copied().unwrap_or_default();
total_count - self.initial_count
}
}

#[derive(Clone)]
Expand Down Expand Up @@ -309,6 +360,7 @@ impl ObjectClient for MockClient {
key: &str,
) -> ObjectClientResult<DeleteObjectResult, DeleteObjectError, Self::ClientError> {
trace!(bucket, key, "DeleteObject");
self.inc_op_count(Operation::DeleteObject);

if bucket != self.config.bucket {
return Err(ObjectClientError::ServiceError(DeleteObjectError::NoSuchBucket));
Expand All @@ -327,6 +379,7 @@ impl ObjectClient for MockClient {
if_match: Option<ETag>,
) -> ObjectClientResult<Self::GetObjectResult, GetObjectError, Self::ClientError> {
trace!(bucket, key, ?range, ?if_match, "GetObject");
self.inc_op_count(Operation::GetObject);

if bucket != self.config.bucket {
return Err(ObjectClientError::ServiceError(GetObjectError::NoSuchBucket));
Expand Down Expand Up @@ -367,6 +420,7 @@ impl ObjectClient for MockClient {
key: &str,
) -> ObjectClientResult<HeadObjectResult, HeadObjectError, Self::ClientError> {
trace!(bucket, key, "HeadObject");
self.inc_op_count(Operation::HeadObject);

if bucket != self.config.bucket {
return Err(ObjectClientError::ServiceError(HeadObjectError::NotFound));
Expand Down Expand Up @@ -399,6 +453,7 @@ impl ObjectClient for MockClient {
prefix: &str,
) -> ObjectClientResult<ListObjectsResult, ListObjectsError, Self::ClientError> {
trace!(bucket, ?continuation_token, delimiter, max_keys, prefix, "ListObjects");
self.inc_op_count(Operation::ListObjectsV2);

if bucket != self.config.bucket {
return Err(ObjectClientError::ServiceError(ListObjectsError::NoSuchBucket));
Expand Down Expand Up @@ -499,6 +554,7 @@ impl ObjectClient for MockClient {
params: &PutObjectParams,
) -> ObjectClientResult<Self::PutObjectRequest, PutObjectError, Self::ClientError> {
trace!(bucket, key, "PutObject");
self.inc_op_count(Operation::PutObject);

if bucket != self.config.bucket {
return Err(ObjectClientError::ServiceError(PutObjectError::NoSuchBucket));
Expand All @@ -523,6 +579,7 @@ impl ObjectClient for MockClient {
object_attributes: &[ObjectAttribute],
) -> ObjectClientResult<GetObjectAttributesResult, GetObjectAttributesError, Self::ClientError> {
trace!(bucket, key, "GetObjectAttributes");
self.inc_op_count(Operation::GetObjectAttributes);

if bucket != self.config.bucket {
return Err(ObjectClientError::ServiceError(GetObjectAttributesError::NoSuchBucket));
Expand Down Expand Up @@ -960,4 +1017,31 @@ mod tests {
matches!(&list_result.objects[..], [object] if object.key == key && object.storage_class.as_deref() == storage_class )
);
}

#[tokio::test]
async fn counter_test() {
let bucket = "test_bucket";
let client = MockClient::new(MockClientConfig {
bucket: bucket.to_owned(),
part_size: 1024,
});

let head_counter_1 = client.new_counter(Operation::HeadObject);
let delete_counter_1 = client.new_counter(Operation::DeleteObject);

let _result = client.head_object(bucket, "key").await;
assert_eq!(1, head_counter_1.count());
assert_eq!(0, delete_counter_1.count());

let head_counter_2 = client.new_counter(Operation::HeadObject);
assert_eq!(0, head_counter_2.count());

let _result = client.head_object(bucket, "key").await;
let _result = client.delete_object(bucket, "key").await;
let _result = client.delete_object(bucket, "key").await;
let _result = client.delete_object(bucket, "key").await;
assert_eq!(2, head_counter_1.count());
assert_eq!(3, delete_counter_1.count());
assert_eq!(1, head_counter_2.count());
}
}
Loading

0 comments on commit fa0d516

Please sign in to comment.