From 035b3dc48a26ac1350c0ba12872f9d97d602840c Mon Sep 17 00:00:00 2001 From: Weny Xu Date: Sun, 31 Dec 2023 14:51:02 +0900 Subject: [PATCH] feat: add behavior tests for blocking buffer reader (#3872) --- core/src/types/operator/operator_functions.rs | 6 + core/tests/behavior/blocking_write.rs | 123 ++++++++++++++++++ 2 files changed, 129 insertions(+) diff --git a/core/src/types/operator/operator_functions.rs b/core/src/types/operator/operator_functions.rs index 4afa157f02e8..07644e2089e5 100644 --- a/core/src/types/operator/operator_functions.rs +++ b/core/src/types/operator/operator_functions.rs @@ -395,6 +395,12 @@ impl FunctionReader { pub fn call(self) -> Result { self.0.call() } + + /// Set the buffer capability to enable `BufferReader`. + pub fn buffer(mut self, cap: usize) -> Self { + self.0 = self.0.map_args(|args| args.with_buffer(cap)); + self + } } /// Function that generated by [`BlockingOperator::stat_with`]. diff --git a/core/tests/behavior/blocking_write.rs b/core/tests/behavior/blocking_write.rs index bb1e7e531a31..3b890655ab0c 100644 --- a/core/tests/behavior/blocking_write.rs +++ b/core/tests/behavior/blocking_write.rs @@ -49,8 +49,11 @@ pub fn behavior_blocking_write_tests(op: &Operator) -> Vec { test_blocking_read_large_range, test_blocking_read_not_exist, test_blocking_fuzz_range_reader, + test_blocking_fuzz_range_reader_with_buffer, test_blocking_fuzz_offset_reader, + test_blocking_fuzz_offset_reader_with_buffer, test_blocking_fuzz_part_reader, + test_blocking_fuzz_part_reader_with_buffer, test_blocking_delete_file, test_blocking_remove_one_file ) @@ -347,6 +350,47 @@ pub fn test_blocking_fuzz_range_reader(op: BlockingOperator) -> Result<()> { Ok(()) } +pub fn test_blocking_fuzz_range_reader_with_buffer(op: BlockingOperator) -> Result<()> { + if !op.info().full_capability().read_with_range { + return Ok(()); + } + + let path = uuid::Uuid::new_v4().to_string(); + debug!("Generate a random file: {}", &path); + let (content, _) = gen_bytes(op.info().full_capability()); + + op.write(&path, content.clone()) + .expect("write must succeed"); + + let mut fuzzer = ObjectReaderFuzzer::new(&path, content.clone(), 0, content.len()); + let mut o = op + .reader_with(&path) + .range(0..content.len() as u64) + .buffer(4096) + .call()?; + + for _ in 0..100 { + match fuzzer.fuzz() { + ObjectReaderAction::Read(size) => { + let mut bs = vec![0; size]; + let n = o.read(&mut bs)?; + fuzzer.check_read(n, &bs[..n]) + } + ObjectReaderAction::Seek(input_pos) => { + let actual_pos = o.seek(input_pos)?; + fuzzer.check_seek(input_pos, actual_pos) + } + ObjectReaderAction::Next => { + let actual_bs = o.next().map(|v| v.expect("next should not return error")); + fuzzer.check_next(actual_bs) + } + } + } + + op.delete(&path).expect("delete must succeed"); + Ok(()) +} + pub fn test_blocking_fuzz_offset_reader(op: BlockingOperator) -> Result<()> { if !op.info().full_capability().read_with_range { return Ok(()); @@ -384,6 +428,43 @@ pub fn test_blocking_fuzz_offset_reader(op: BlockingOperator) -> Result<()> { Ok(()) } +pub fn test_blocking_fuzz_offset_reader_with_buffer(op: BlockingOperator) -> Result<()> { + if !op.info().full_capability().read_with_range { + return Ok(()); + } + + let path = uuid::Uuid::new_v4().to_string(); + debug!("Generate a random file: {}", &path); + let (content, _) = gen_bytes(op.info().full_capability()); + + op.write(&path, content.clone()) + .expect("write must succeed"); + + let mut fuzzer = ObjectReaderFuzzer::new(&path, content.clone(), 0, content.len()); + let mut o = op.reader_with(&path).range(0..).buffer(4096).call()?; + + for _ in 0..100 { + match fuzzer.fuzz() { + ObjectReaderAction::Read(size) => { + let mut bs = vec![0; size]; + let n = o.read(&mut bs)?; + fuzzer.check_read(n, &bs[..n]) + } + ObjectReaderAction::Seek(input_pos) => { + let actual_pos = o.seek(input_pos)?; + fuzzer.check_seek(input_pos, actual_pos) + } + ObjectReaderAction::Next => { + let actual_bs = o.next().map(|v| v.expect("next should not return error")); + fuzzer.check_next(actual_bs) + } + } + } + + op.delete(&path).expect("delete must succeed"); + Ok(()) +} + pub fn test_blocking_fuzz_part_reader(op: BlockingOperator) -> Result<()> { if !op.info().full_capability().read_with_range { return Ok(()); @@ -425,6 +506,48 @@ pub fn test_blocking_fuzz_part_reader(op: BlockingOperator) -> Result<()> { Ok(()) } +pub fn test_blocking_fuzz_part_reader_with_buffer(op: BlockingOperator) -> Result<()> { + if !op.info().full_capability().read_with_range { + return Ok(()); + } + + let path = uuid::Uuid::new_v4().to_string(); + debug!("Generate a random file: {}", &path); + let (content, size) = gen_bytes(op.info().full_capability()); + let (offset, length) = gen_offset_length(size); + + op.write(&path, content.clone()) + .expect("write must succeed"); + + let mut fuzzer = ObjectReaderFuzzer::new(&path, content, offset as usize, length as usize); + let mut o = op + .reader_with(&path) + .range(offset..offset + length) + .buffer(4096) + .call()?; + + for _ in 0..100 { + match fuzzer.fuzz() { + ObjectReaderAction::Read(size) => { + let mut bs = vec![0; size]; + let n = o.read(&mut bs)?; + fuzzer.check_read(n, &bs[..n]) + } + ObjectReaderAction::Seek(input_pos) => { + let actual_pos = o.seek(input_pos)?; + fuzzer.check_seek(input_pos, actual_pos) + } + ObjectReaderAction::Next => { + let actual_bs = o.next().map(|v| v.expect("next should not return error")); + fuzzer.check_next(actual_bs) + } + } + } + + op.delete(&path).expect("delete must succeed"); + Ok(()) +} + // Delete existing file should succeed. pub fn test_blocking_delete_file(op: BlockingOperator) -> Result<()> { let path = uuid::Uuid::new_v4().to_string();