use clap::{Arg, ArgAction, Command};
use fuser::{BackgroundSession, MountOption, Session};
use mountpoint_s3::fuse::S3FuseFilesystem;
use mountpoint_s3::S3FilesystemConfig;
use mountpoint_s3_client::{S3ClientConfig, S3CrtClient};
use mountpoint_s3_crt::common::rust_log_adapter::RustLogAdapter;
use std::{
fs::File,
fs::OpenOptions,
io::{self, BufRead, BufReader},
time::Instant,
};
use tempfile::tempdir;
use tracing_subscriber::fmt::Subscriber;
use tracing_subscriber::util::SubscriberInitExt;
use tracing_subscriber::EnvFilter;
#[cfg(target_os = "linux")]
use std::os::unix::fs::OpenOptionsExt;
fn init_tracing_subscriber() {
RustLogAdapter::try_init().expect("unable to install CRT log adapter");
let subscriber = Subscriber::builder()
.with_env_filter(EnvFilter::from_default_env())
.with_writer(std::io::stderr)
.finish();
subscriber.try_init().expect("unable to install global subscriber");
}
fn main() -> io::Result<()> {
init_tracing_subscriber();
const KB: usize = 1 << 10;
const MB: usize = 1 << 20;
const DEFAULT_BUF_CAP: usize = 128;
let matches = Command::new("benchmark")
.about("Read a single file from a path and ignore its contents")
.arg(Arg::new("bucket").required(true))
.arg(
Arg::new("file_path")
.required(true)
.help("relative path to the mountpoint"),
)
.arg(
Arg::new("buffer-capacity-kb")
.long("buffer-capacity-kb")
.help("Buffer reader capacity in KB"),
)
.arg(
Arg::new("direct")
.long("direct")
.help("Open file with O_DIRECT option")
.action(ArgAction::SetTrue),
)
.arg(
Arg::new("throughput-target-gbps")
.long("throughput-target-gbps")
.help("Desired throughput in Gbps"),
)
.arg(
Arg::new("iterations")
.long("iterations")
.help("Number of times to download"),
)
.arg(Arg::new("region").long("region").default_value("us-east-1"))
.get_matches();
let bucket_name = matches.get_one::<String>("bucket").unwrap();
let file_path = matches.get_one::<String>("file_path").unwrap();
let buffer_capacity = matches
.get_one::<String>("buffer-capacity-kb")
.map(|s| s.parse::<usize>().expect("buffer capacity must be a number"));
let direct = matches.get_flag("direct");
let throughput_target_gbps = matches
.get_one::<String>("throughput-target-gbps")
.map(|s| s.parse::<f64>().expect("throughput target must be an f64"));
let iterations = matches
.get_one::<String>("iterations")
.map(|s| s.parse::<usize>().expect("iterations must be a number"));
let region = matches.get_one::<String>("region").unwrap();
let session = mount_file_system(bucket_name, region, throughput_target_gbps);
let mountpoint = &session.mountpoint;
#[cfg(not(target_os = "linux"))]
if direct {
panic!("O_DIRECT only supported on Linux");
}
for i in 0..iterations.unwrap_or(1) {
let file_path = mountpoint.join(file_path);
let file = if direct {
let mut open = OpenOptions::new();
open.read(true);
#[cfg(target_os = "linux")]
open.custom_flags(libc::O_DIRECT);
open.open(file_path)?
} else {
File::open(file_path)?
};
let mut received_size: u64 = 0;
let mut op_counter: u64 = 0;
let start = Instant::now();
let mut reader = BufReader::with_capacity(buffer_capacity.unwrap_or(DEFAULT_BUF_CAP) * KB, file);
loop {
let length = {
let buffer = reader.fill_buf()?;
op_counter += 1;
received_size += buffer.len() as u64;
buffer.len()
};
if length == 0 {
break;
}
reader.consume(length);
}
let elapsed = start.elapsed();
println!(
"{}: requested {} ops in {:.2}s: {:.2} IOPS",
i,
op_counter,
elapsed.as_secs_f64(),
(op_counter as f64) / elapsed.as_secs_f64()
);
println!(
"{}: received {} bytes in {:.2}s: {:.2} MiB/s",
i,
received_size,
elapsed.as_secs_f64(),
(received_size as f64) / elapsed.as_secs_f64() / MB as f64
);
}
drop(session);
Ok(())
}
fn mount_file_system(bucket_name: &str, region: &str, throughput_target_gbps: Option<f64>) -> BackgroundSession {
let temp_dir = tempdir().expect("Should be able to create temp directory");
let mountpoint = temp_dir.path();
let config = S3ClientConfig {
throughput_target_gbps,
..Default::default()
};
let client = S3CrtClient::new(region, config).expect("Failed to create S3 client");
let runtime = client.event_loop_group();
let mut options = vec![MountOption::RO, MountOption::FSName("mountpoint-s3".to_string())];
options.push(MountOption::AutoUnmount);
let filesystem_config = S3FilesystemConfig::default();
println!(
"Mounting bucket {} to path {}",
bucket_name,
mountpoint.to_str().unwrap()
);
let session = Session::new(
S3FuseFilesystem::new(client, runtime, bucket_name, &Default::default(), filesystem_config),
mountpoint,
&options,
)
.expect("Should have created FUSE session successfully");
BackgroundSession::new(session).expect("Should have started FUSE session successfully")
}
来源:
标签:文件,use,读取,aws,let,expect,file,mountpoint,target From: https://www.cnblogs.com/itfanr/p/17340761.htmlhttps://github.com/awslabs/mountpoint-s3/blob/main/mountpoint-s3/examples/fs_benchmark.rs