ella 0.1.5

A streaming time-series datastore for low-latency applications.
Documentation
use std::future::IntoFuture;

use ella_engine::registry::Id;
use futures::{future::BoxFuture, FutureExt};

use crate::Ella;

#[must_use]
#[derive(Debug)]
pub struct GetCatalog<'a> {
    inner: &'a Ella,
    catalog: Id<'a>,
}

impl<'a> GetCatalog<'a> {
    pub(crate) fn new(inner: &'a Ella, catalog: Id<'a>) -> Self {
        Self { inner, catalog }
    }

    pub fn or_create(self) -> GetOrCreateCatalog<'a> {
        GetOrCreateCatalog {
            inner: self.inner,
            catalog: self.catalog,
        }
    }

    pub fn and_use(self) -> UseCatalog<'a> {
        UseCatalog {
            inner: self.inner,
            catalog: self.catalog,
            create: false,
        }
    }

    pub fn drop(self) -> DropCatalog<'a> {
        DropCatalog {
            inner: self.inner,
            catalog: self.catalog,
            if_exists: false,
            cascade: false,
        }
    }
}

#[must_use]
#[derive(Debug)]
pub struct GetOrCreateCatalog<'a> {
    inner: &'a Ella,
    catalog: Id<'a>,
}

impl<'a> GetOrCreateCatalog<'a> {
    pub fn and_use(self) -> UseCatalog<'a> {
        UseCatalog {
            inner: self.inner,
            catalog: self.catalog,
            create: true,
        }
    }
}

#[must_use]
#[derive(Debug)]
pub struct UseCatalog<'a> {
    inner: &'a Ella,
    catalog: Id<'a>,
    create: bool,
}

impl<'a> IntoFuture for UseCatalog<'a> {
    type Output = crate::Result<Ella>;
    type IntoFuture = BoxFuture<'a, Self::Output>;

    fn into_future(self) -> Self::IntoFuture {
        async move {
            let mut this = self.inner.clone();
            if self.create {
                this.create_catalog(self.catalog.clone(), true).await?;
            }
            this.use_catalog(self.catalog).await
        }
        .boxed()
    }
}

#[must_use]
#[derive(Debug)]
pub struct DropCatalog<'a> {
    inner: &'a Ella,
    catalog: Id<'a>,
    if_exists: bool,
    cascade: bool,
}

impl<'a> DropCatalog<'a> {
    pub fn if_exists(mut self) -> Self {
        self.if_exists = true;
        self
    }

    pub fn cascade(mut self) -> Self {
        self.cascade = true;
        self
    }
}

impl<'a> IntoFuture for DropCatalog<'a> {
    type Output = crate::Result<&'a Ella>;
    type IntoFuture = BoxFuture<'a, Self::Output>;

    fn into_future(self) -> Self::IntoFuture {
        async move {
            let if_exists = if self.if_exists { "IF EXISTS " } else { "" };
            let cascade = if self.cascade { " CASCADE" } else { "" };
            self.inner
                .execute(format!("DROP CATALOG {if_exists}{}{cascade}", self.catalog))
                .await?;
            Ok(self.inner)
        }
        .boxed()
    }
}