saola-quaint 0.2.0-alpha.14

An abstraction layer for SQL databases (PostgreSQL, MySQL, SQLite, MSSQL)
use super::TestApi;
use crate::{connector::Queryable, single::Quaint};
use names::Generator;
use quaint_test_setup::Tags;
use std::{env, sync::LazyLock};

pub static CONN_STR: LazyLock<String> = LazyLock::new(|| env::var("TEST_MYSQL").expect("TEST_MYSQL env var"));
pub static CONN_STR8: LazyLock<String> = LazyLock::new(|| env::var("TEST_MYSQL8").expect("TEST_MYSQL8 env var"));
pub static CONN_STR_MARIADB: LazyLock<String> =
    LazyLock::new(|| env::var("TEST_MYSQL_MARIADB").expect("TEST_MYSQL_MARIADB env var"));

pub(crate) async fn mysql_test_api<'a>() -> crate::Result<MySql<'a>> {
    MySql::new(CONN_STR.as_str(), Tags::MYSQL5_7).await
}

pub(crate) async fn mysql5_7_test_api<'a>() -> crate::Result<MySql<'a>> {
    MySql::new(CONN_STR.as_str(), Tags::MYSQL5_7).await
}

pub(crate) async fn mysql8_test_api<'a>() -> crate::Result<MySql<'a>> {
    MySql::new(CONN_STR8.as_str(), Tags::MYSQL8).await
}

pub(crate) async fn mysql_mariadb_test_api<'a>() -> crate::Result<MySql<'a>> {
    MySql::new(CONN_STR_MARIADB.as_str(), Tags::MYSQL_MARIADB).await
}

pub struct MySql<'a> {
    names: Generator<'a>,
    conn: Quaint,
    conn_str: String,
    tag: Tags,
}

impl<'a> MySql<'a> {
    pub async fn new(conn_str: &str, tag: Tags) -> crate::Result<MySql<'a>> {
        let names = Generator::default();
        let conn = Quaint::new(conn_str).await?;

        Ok(Self {
            names,
            conn,
            tag,
            conn_str: conn_str.into(),
        })
    }

    fn render_perm_create_table(&mut self, table_name: &str, columns: &str) -> (String, String) {
        let create = format!(
            r##"
            CREATE TABLE `{table_name}` ({columns}) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci
            "##,
        );

        (table_name.to_string(), create)
    }
}

#[async_trait::async_trait]
impl TestApi for MySql<'_> {
    fn system(&self) -> &'static str {
        "mysql"
    }

    async fn create_type_table(&mut self, r#type: &str) -> crate::Result<String> {
        self.create_temp_table(&format!("{}, `value` {}", self.autogen_id("id"), r#type))
            .await
    }

    async fn create_temp_table(&mut self, columns: &str) -> crate::Result<String> {
        let name = self.get_name();

        let (name, create) = self.render_create_table(&name, columns);

        self.conn().raw_cmd(&create).await?;

        Ok(name)
    }

    async fn create_table(&mut self, columns: &str) -> crate::Result<String> {
        let name = self.get_name();

        let (name, create) = self.render_perm_create_table(&name, columns);

        self.conn().raw_cmd(&create).await?;

        Ok(name)
    }

    async fn delete_table(&mut self, table_name: &str) -> crate::Result<()> {
        let delete = format!(
            r##"
            DROP TABLE `{table_name}`
            "##
        );
        self.conn().raw_cmd(&delete).await
    }

    fn render_create_table(&mut self, table_name: &str, columns: &str) -> (String, String) {
        let create = format!(
            r##"
            CREATE TEMPORARY TABLE `{table_name}` ({columns}) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci
            "##,
        );

        (table_name.to_string(), create)
    }

    async fn create_index(&mut self, table: &str, columns: &str) -> crate::Result<String> {
        let name = self.get_name();

        let create = format!(
            r##"
            CREATE UNIQUE INDEX {name} ON {table} ({columns})
            "##
        );

        self.conn().raw_cmd(&create).await?;

        Ok(name)
    }

    fn conn(&self) -> &Quaint {
        &self.conn
    }

    async fn create_additional_connection(&self) -> crate::Result<Quaint> {
        Quaint::new(&self.conn_str).await
    }

    fn create_pool(&self) -> crate::Result<crate::pooled::Quaint> {
        Ok(crate::pooled::Quaint::builder(&CONN_STR)?.build())
    }

    fn unique_constraint(&mut self, column: &str) -> String {
        format!("UNIQUE({column})")
    }

    fn foreign_key(&mut self, parent_table: &str, parent_column: &str, child_column: &str) -> String {
        let name = self.get_name();

        format!(
            "CONSTRAINT {} FOREIGN KEY ({}) REFERENCES {}({})",
            &name, child_column, parent_table, parent_column
        )
    }

    fn autogen_id(&self, name: &str) -> String {
        format!("{name} INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY")
    }

    fn get_name(&mut self) -> String {
        self.names.next().unwrap().replace('-', "")
    }

    fn connector_tag(&self) -> Tags {
        self.tag
    }
}