2022-12-30 08:23:19 +00:00
|
|
|
use crate::api::conn::Method;
|
|
|
|
use crate::api::conn::Param;
|
|
|
|
use crate::api::conn::Router;
|
|
|
|
use crate::api::Connection;
|
|
|
|
use crate::api::Error;
|
|
|
|
use crate::api::ExtraFeatures;
|
|
|
|
use crate::api::Result;
|
2023-08-31 19:44:18 +00:00
|
|
|
use crate::opt::ExportDestination;
|
|
|
|
use channel::Receiver;
|
|
|
|
use futures::Stream;
|
|
|
|
use futures::StreamExt;
|
2022-12-30 08:23:19 +00:00
|
|
|
use std::future::Future;
|
|
|
|
use std::future::IntoFuture;
|
2023-08-31 19:44:18 +00:00
|
|
|
use std::marker::PhantomData;
|
2022-12-30 08:23:19 +00:00
|
|
|
use std::path::PathBuf;
|
|
|
|
use std::pin::Pin;
|
2023-08-31 19:44:18 +00:00
|
|
|
use std::task::Context;
|
|
|
|
use std::task::Poll;
|
2022-12-30 08:23:19 +00:00
|
|
|
|
|
|
|
/// A database export future
|
|
|
|
#[derive(Debug)]
|
2023-04-19 08:26:22 +00:00
|
|
|
#[must_use = "futures do nothing unless you `.await` or poll them"]
|
2023-08-31 19:44:18 +00:00
|
|
|
pub struct Export<'r, C: Connection, R> {
|
2022-12-30 08:23:19 +00:00
|
|
|
pub(super) router: Result<&'r Router<C>>,
|
2023-08-31 19:44:18 +00:00
|
|
|
pub(super) target: ExportDestination,
|
|
|
|
pub(super) response: PhantomData<R>,
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
|
2023-08-31 19:44:18 +00:00
|
|
|
impl<'r, Client> IntoFuture for Export<'r, Client, PathBuf>
|
2022-12-30 08:23:19 +00:00
|
|
|
where
|
|
|
|
Client: Connection,
|
|
|
|
{
|
|
|
|
type Output = Result<()>;
|
|
|
|
type IntoFuture = Pin<Box<dyn Future<Output = Self::Output> + Send + Sync + 'r>>;
|
|
|
|
|
|
|
|
fn into_future(self) -> Self::IntoFuture {
|
|
|
|
Box::pin(async {
|
|
|
|
let router = self.router?;
|
|
|
|
if !router.features.contains(&ExtraFeatures::Backup) {
|
|
|
|
return Err(Error::BackupsNotSupported.into());
|
|
|
|
}
|
|
|
|
let mut conn = Client::new(Method::Export);
|
2023-08-09 12:15:20 +00:00
|
|
|
match self.target {
|
2023-08-31 19:44:18 +00:00
|
|
|
ExportDestination::File(path) => conn.execute_unit(router, Param::file(path)).await,
|
|
|
|
ExportDestination::Memory => unreachable!(),
|
2023-08-09 12:15:20 +00:00
|
|
|
}
|
2022-12-30 08:23:19 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2023-08-09 12:15:20 +00:00
|
|
|
|
2023-08-31 19:44:18 +00:00
|
|
|
impl<'r, Client> IntoFuture for Export<'r, Client, ()>
|
|
|
|
where
|
|
|
|
Client: Connection,
|
|
|
|
{
|
|
|
|
type Output = Result<Backup>;
|
|
|
|
type IntoFuture = Pin<Box<dyn Future<Output = Self::Output> + Send + Sync + 'r>>;
|
2023-08-09 12:15:20 +00:00
|
|
|
|
2023-08-31 19:44:18 +00:00
|
|
|
fn into_future(self) -> Self::IntoFuture {
|
|
|
|
Box::pin(async move {
|
|
|
|
let router = self.router?;
|
|
|
|
if !router.features.contains(&ExtraFeatures::Backup) {
|
|
|
|
return Err(Error::BackupsNotSupported.into());
|
|
|
|
}
|
|
|
|
let (tx, rx) = crate::channel::new(1);
|
|
|
|
let mut conn = Client::new(Method::Export);
|
|
|
|
let ExportDestination::Memory = self.target else {
|
|
|
|
unreachable!();
|
|
|
|
};
|
|
|
|
conn.execute_unit(router, Param::sender(tx)).await?;
|
|
|
|
Ok(Backup {
|
|
|
|
rx,
|
|
|
|
})
|
|
|
|
})
|
2023-08-09 12:15:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-31 19:44:18 +00:00
|
|
|
/// A stream of exported data
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
#[must_use = "streams do nothing unless you poll them"]
|
|
|
|
pub struct Backup {
|
|
|
|
rx: Receiver<Result<Vec<u8>>>,
|
2023-08-09 12:15:20 +00:00
|
|
|
}
|
|
|
|
|
2023-08-31 19:44:18 +00:00
|
|
|
impl Stream for Backup {
|
|
|
|
type Item = Result<Vec<u8>>;
|
2023-08-09 12:15:20 +00:00
|
|
|
|
2023-08-31 19:44:18 +00:00
|
|
|
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
|
|
|
self.as_mut().rx.poll_next_unpin(cx)
|
2023-08-09 12:15:20 +00:00
|
|
|
}
|
|
|
|
}
|