use std::sync::Arc;
use std::vec::IntoIter;
use super::Adapter;
use crate::raw::oio::HierarchyLister;
use crate::raw::oio::QueueBuf;
use crate::raw::*;
use crate::*;
#[derive(Debug, Clone)]
pub struct Backend<S: Adapter> {
kv: Arc<S>,
root: String,
}
impl<S> Backend<S>
where
S: Adapter,
{
pub fn new(kv: S) -> Self {
Self {
kv: Arc::new(kv),
root: "/".to_string(),
}
}
pub fn with_root(self, root: &str) -> Self {
self.with_normalized_root(normalize_root(root))
}
pub(crate) fn with_normalized_root(mut self, root: String) -> Self {
self.root = root;
self
}
}
impl<S: Adapter> Access for Backend<S> {
type Reader = Buffer;
type BlockingReader = Buffer;
type Writer = KvWriter<S>;
type BlockingWriter = KvWriter<S>;
type Lister = HierarchyLister<KvLister>;
type BlockingLister = HierarchyLister<KvLister>;
fn info(&self) -> Arc<AccessorInfo> {
let mut am: AccessorInfo = self.kv.metadata().into();
am.set_root(&self.root);
let mut cap = am.native_capability();
if cap.read {
cap.stat = true;
}
if cap.write {
cap.write_can_empty = true;
cap.delete = true;
}
if cap.list {
cap.list_with_recursive = true;
}
am.set_native_capability(cap);
am.into()
}
async fn read(&self, path: &str, args: OpRead) -> Result<(RpRead, Self::Reader)> {
let p = build_abs_path(&self.root, path);
let bs = match self.kv.get(&p).await? {
Some(bs) => bs,
None => return Err(Error::new(ErrorKind::NotFound, "kv doesn't have this path")),
};
Ok((RpRead::new(), bs.slice(args.range().to_range_as_usize())))
}
fn blocking_read(&self, path: &str, args: OpRead) -> Result<(RpRead, Self::BlockingReader)> {
let p = build_abs_path(&self.root, path);
let bs = match self.kv.blocking_get(&p)? {
Some(bs) => bs,
None => return Err(Error::new(ErrorKind::NotFound, "kv doesn't have this path")),
};
Ok((RpRead::new(), bs.slice(args.range().to_range_as_usize())))
}
async fn write(&self, path: &str, _: OpWrite) -> Result<(RpWrite, Self::Writer)> {
let p = build_abs_path(&self.root, path);
Ok((RpWrite::new(), KvWriter::new(self.kv.clone(), p)))
}
fn blocking_write(&self, path: &str, _: OpWrite) -> Result<(RpWrite, Self::BlockingWriter)> {
let p = build_abs_path(&self.root, path);
Ok((RpWrite::new(), KvWriter::new(self.kv.clone(), p)))
}
async fn stat(&self, path: &str, _: OpStat) -> Result<RpStat> {
let p = build_abs_path(&self.root, path);
if p == build_abs_path(&self.root, "") {
Ok(RpStat::new(Metadata::new(EntryMode::DIR)))
} else {
let bs = self.kv.get(&p).await?;
match bs {
Some(bs) => Ok(RpStat::new(
Metadata::new(EntryMode::FILE).with_content_length(bs.len() as u64),
)),
None => Err(Error::new(ErrorKind::NotFound, "kv doesn't have this path")),
}
}
}
fn blocking_stat(&self, path: &str, _: OpStat) -> Result<RpStat> {
let p = build_abs_path(&self.root, path);
if p == build_abs_path(&self.root, "") {
Ok(RpStat::new(Metadata::new(EntryMode::DIR)))
} else {
let bs = self.kv.blocking_get(&p)?;
match bs {
Some(bs) => Ok(RpStat::new(
Metadata::new(EntryMode::FILE).with_content_length(bs.len() as u64),
)),
None => Err(Error::new(ErrorKind::NotFound, "kv doesn't have this path")),
}
}
}
async fn delete(&self, path: &str, _: OpDelete) -> Result<RpDelete> {
let p = build_abs_path(&self.root, path);
self.kv.delete(&p).await?;
Ok(RpDelete::default())
}
fn blocking_delete(&self, path: &str, _: OpDelete) -> Result<RpDelete> {
let p = build_abs_path(&self.root, path);
self.kv.blocking_delete(&p)?;
Ok(RpDelete::default())
}
async fn list(&self, path: &str, args: OpList) -> Result<(RpList, Self::Lister)> {
let p = build_abs_path(&self.root, path);
let res = self.kv.scan(&p).await?;
let lister = KvLister::new(&self.root, res);
let lister = HierarchyLister::new(lister, path, args.recursive());
Ok((RpList::default(), lister))
}
fn blocking_list(&self, path: &str, args: OpList) -> Result<(RpList, Self::BlockingLister)> {
let p = build_abs_path(&self.root, path);
let res = self.kv.blocking_scan(&p)?;
let lister = KvLister::new(&self.root, res);
let lister = HierarchyLister::new(lister, path, args.recursive());
Ok((RpList::default(), lister))
}
}
pub struct KvLister {
root: String,
inner: IntoIter<String>,
}
impl KvLister {
fn new(root: &str, inner: Vec<String>) -> Self {
Self {
root: root.to_string(),
inner: inner.into_iter(),
}
}
fn inner_next(&mut self) -> Option<oio::Entry> {
self.inner.next().map(|v| {
let mode = if v.ends_with('/') {
EntryMode::DIR
} else {
EntryMode::FILE
};
let mut path = build_rel_path(&self.root, &v);
if path.is_empty() {
path = "/".to_string();
}
oio::Entry::new(&path, Metadata::new(mode))
})
}
}
impl oio::List for KvLister {
async fn next(&mut self) -> Result<Option<oio::Entry>> {
Ok(self.inner_next())
}
}
impl oio::BlockingList for KvLister {
fn next(&mut self) -> Result<Option<oio::Entry>> {
Ok(self.inner_next())
}
}
pub struct KvWriter<S> {
kv: Arc<S>,
path: String,
buffer: QueueBuf,
}
impl<S> KvWriter<S> {
fn new(kv: Arc<S>, path: String) -> Self {
KvWriter {
kv,
path,
buffer: QueueBuf::new(),
}
}
}
unsafe impl<S: Adapter> Sync for KvWriter<S> {}
impl<S: Adapter> oio::Write for KvWriter<S> {
async fn write(&mut self, bs: Buffer) -> Result<()> {
self.buffer.push(bs);
Ok(())
}
async fn close(&mut self) -> Result<()> {
let buf = self.buffer.clone().collect();
self.kv.set(&self.path, buf).await
}
async fn abort(&mut self) -> Result<()> {
self.buffer.clear();
Ok(())
}
}
impl<S: Adapter> oio::BlockingWrite for KvWriter<S> {
fn write(&mut self, bs: Buffer) -> Result<()> {
self.buffer.push(bs);
Ok(())
}
fn close(&mut self) -> Result<()> {
let buf = self.buffer.clone().collect();
self.kv.blocking_set(&self.path, buf)?;
Ok(())
}
}