use std::fmt::Debug;
use std::fmt::Formatter;
use std::sync::Arc;
use crate::raw::oio::FlatLister;
use crate::raw::oio::List;
use crate::raw::oio::PrefixLister;
use crate::raw::*;
use crate::*;
#[derive(Debug, Clone)]
pub struct SimulateLayer {
list_recursive: bool,
stat_dir: bool,
create_dir: bool,
delete_recursive: bool,
}
impl Default for SimulateLayer {
fn default() -> Self {
Self {
list_recursive: true,
stat_dir: true,
create_dir: true,
delete_recursive: true,
}
}
}
impl SimulateLayer {
pub fn with_list_recursive(mut self, enabled: bool) -> Self {
self.list_recursive = enabled;
self
}
pub fn with_stat_dir(mut self, enabled: bool) -> Self {
self.stat_dir = enabled;
self
}
pub fn with_create_dir(mut self, enabled: bool) -> Self {
self.create_dir = enabled;
self
}
pub fn with_delete_recursive(mut self, enabled: bool) -> Self {
self.delete_recursive = enabled;
self
}
}
impl<A: Access> Layer<A> for SimulateLayer {
type LayeredAccess = SimulateAccessor<A>;
fn layer(&self, inner: A) -> Self::LayeredAccess {
let info = inner.info();
info.update_full_capability(|mut cap| {
if self.create_dir && cap.list && cap.write_can_empty {
cap.create_dir = true;
}
if self.delete_recursive && cap.list && cap.delete {
cap.delete_with_recursive = true;
}
cap
});
SimulateAccessor {
config: self.clone(),
info,
inner: Arc::new(inner),
}
}
}
pub struct SimulateAccessor<A: Access> {
config: SimulateLayer,
info: Arc<AccessorInfo>,
inner: Arc<A>,
}
impl<A: Access> Debug for SimulateAccessor<A> {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
self.inner.fmt(f)
}
}
impl<A: Access> SimulateAccessor<A> {
async fn simulate_create_dir(&self, path: &str, args: OpCreateDir) -> Result<RpCreateDir> {
let capability = self.info.native_capability();
if capability.create_dir || !self.config.create_dir {
return self.inner().create_dir(path, args).await;
}
if capability.write_can_empty && capability.list {
let (_, mut w) = self.inner.write(path, OpWrite::default()).await?;
oio::Write::close(&mut w).await?;
return Ok(RpCreateDir::default());
}
self.inner.create_dir(path, args).await
}
async fn simulate_stat(&self, path: &str, args: OpStat) -> Result<RpStat> {
let capability = self.info.native_capability();
if path == "/" {
return Ok(RpStat::new(Metadata::new(EntryMode::DIR)));
}
if path.ends_with('/') {
if capability.create_dir {
let meta = self.inner.stat(path, args.clone()).await?.into_metadata();
if meta.is_file() {
return Err(Error::new(
ErrorKind::NotFound,
"stat expected a directory, but found a file",
));
}
return Ok(RpStat::new(meta));
}
if self.config.stat_dir && capability.list_with_recursive {
let (_, mut l) = self
.inner
.list(path, OpList::default().with_recursive(true).with_limit(1))
.await?;
return if oio::List::next(&mut l).await?.is_some() {
Ok(RpStat::new(Metadata::new(EntryMode::DIR)))
} else {
Err(Error::new(
ErrorKind::NotFound,
"the directory is not found",
))
};
}
}
self.inner.stat(path, args).await
}
async fn simulate_list(
&self,
path: &str,
args: OpList,
) -> Result<(RpList, SimulateLister<A, A::Lister>)> {
let cap = self.info.native_capability();
let recursive = args.recursive();
let forward = args;
let (rp, lister) = match (
recursive,
cap.list_with_recursive,
self.config.list_recursive,
) {
(_, true, _) => {
let (rp, p) = self.inner.list(path, forward).await?;
(rp, SimulateLister::One(p))
}
(true, false, true) => {
if path.ends_with('/') {
let p = FlatLister::new(self.inner.clone(), path);
(RpList::default(), SimulateLister::Two(p))
} else {
let parent = get_parent(path);
let p = FlatLister::new(self.inner.clone(), parent);
let p = PrefixLister::new(p, path);
(RpList::default(), SimulateLister::Four(p))
}
}
(true, false, false) => {
let (rp, p) = self.inner.list(path, forward).await?;
(rp, SimulateLister::One(p))
}
(false, false, _) => {
if path.ends_with('/') {
let (rp, p) = self.inner.list(path, forward).await?;
(rp, SimulateLister::One(p))
} else {
let parent = get_parent(path);
let (rp, p) = self.inner.list(parent, forward).await?;
let p = PrefixLister::new(p, path);
(rp, SimulateLister::Three(p))
}
}
};
Ok((rp, lister))
}
pub(crate) async fn simulate_delete_with_recursive<D: oio::Delete>(
&self,
deleter: &mut D,
path: &str,
args: OpDelete,
) -> Result<()> {
if !self.info.full_capability().delete_with_recursive {
return Err(Error::new(
ErrorKind::Unsupported,
"recursive delete is not supported",
));
}
let non_recursive = args.clone().with_recursive(false);
let (_rp, mut lister) = self
.simulate_list(path, OpList::new().with_recursive(true))
.await?;
while let Some(entry) = lister.next().await? {
let entry = entry.into_entry();
let mut entry_args = non_recursive.clone();
if let Some(version) = entry.metadata().version() {
entry_args = entry_args.with_version(version);
}
deleter.delete(entry.path(), entry_args).await?;
}
Ok(())
}
}
impl<A: Access> LayeredAccess for SimulateAccessor<A> {
type Inner = A;
type Reader = A::Reader;
type Writer = A::Writer;
type Lister = SimulateLister<A, A::Lister>;
type Deleter = SimulateDeleter<A, A::Deleter>;
fn inner(&self) -> &Self::Inner {
&self.inner
}
fn info(&self) -> Arc<AccessorInfo> {
self.info.clone()
}
async fn create_dir(&self, path: &str, args: OpCreateDir) -> Result<RpCreateDir> {
self.simulate_create_dir(path, args).await
}
async fn read(&self, path: &str, args: OpRead) -> Result<(RpRead, Self::Reader)> {
self.inner.read(path, args).await
}
async fn write(&self, path: &str, args: OpWrite) -> Result<(RpWrite, Self::Writer)> {
self.inner.write(path, args).await
}
async fn stat(&self, path: &str, args: OpStat) -> Result<RpStat> {
self.simulate_stat(path, args).await
}
async fn delete(&self) -> Result<(RpDelete, Self::Deleter)> {
let (rp, deleter) = self.inner().delete().await?;
let accessor = SimulateAccessor {
config: self.config.clone(),
info: self.info.clone(),
inner: self.inner.clone(),
};
Ok((rp, SimulateDeleter::new(accessor, deleter)))
}
async fn list(&self, path: &str, args: OpList) -> Result<(RpList, Self::Lister)> {
self.simulate_list(path, args).await
}
async fn presign(&self, path: &str, args: OpPresign) -> Result<RpPresign> {
self.inner.presign(path, args).await
}
}
pub type SimulateLister<A, P> =
FourWays<P, FlatLister<Arc<A>, P>, PrefixLister<P>, PrefixLister<FlatLister<Arc<A>, P>>>;
pub struct SimulateDeleter<A: Access, D> {
accessor: SimulateAccessor<A>,
deleter: D,
}
impl<A: Access, D> SimulateDeleter<A, D> {
pub fn new(accessor: SimulateAccessor<A>, deleter: D) -> Self {
Self { accessor, deleter }
}
}
impl<A: Access, D: oio::Delete> oio::Delete for SimulateDeleter<A, D> {
async fn delete(&mut self, path: &str, args: OpDelete) -> Result<()> {
if args.recursive() {
let cap = self.accessor.info.native_capability();
if cap.delete_with_recursive {
return self.deleter.delete(path, args).await;
}
if self.accessor.config.delete_recursive {
return self
.accessor
.simulate_delete_with_recursive(&mut self.deleter, path, args)
.await;
}
}
self.deleter.delete(path, args).await
}
fn close(&mut self) -> impl Future<Output = Result<()>> + MaybeSend {
self.deleter.close()
}
}