sqlint 0.2.2

An abstraction layer for SQL databases.
Documentation
use super::TestApi;
use crate::{connector::Queryable, single::Sqlint};
use names::Generator;
use once_cell::sync::Lazy;
use std::env;
use test_setup::Tags;

pub static CONN_STR: Lazy<String> = Lazy::new(|| env::var("TEST_MSSQL").expect("TEST_MSSQL env var"));

pub(crate) async fn mssql_test_api<'a>() -> crate::Result<MsSql<'a>> {
    MsSql::new().await
}

pub struct MsSql<'a> {
    names: Generator<'a>,
    conn: Sqlint,
}

impl<'a> MsSql<'a> {
    pub async fn new() -> crate::Result<MsSql<'a>> {
        let names = Generator::default();
        let conn = Sqlint::new(&CONN_STR).await?;

        Ok(Self { names, conn })
    }
}

#[async_trait::async_trait]
impl<'a> TestApi for MsSql<'a> {
    fn system(&self) -> &'static str {
        "mssql"
    }

    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_table(&mut self, _columns: &str) -> crate::Result<String> {
        unimplemented!("only required for MySql nested sub select test")
    }

    async fn delete_table(&mut self, _table_name: &str) -> crate::Result<()> {
        unimplemented!("only required for MySql nested sub select test")
    }

    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_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) -> &Sqlint {
        &self.conn
    }

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

    fn render_create_table(&mut self, table_name: &str, columns: &str) -> (String, String) {
        let table_name = format!("##{table_name}");
        let create = format!(
            r##"
            CREATE TABLE {table_name} ({columns})
            "##,
        );

        (table_name, create)
    }

    fn unique_constraint(&mut self, column: &str) -> String {
        let name = self.names.next().unwrap().replace('-', "");
        format!("CONSTRAINT {name} 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 IDENTITY(1,1) PRIMARY KEY")
    }

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

    fn connector_tag(&self) -> test_setup::Tags {
        Tags::MSSQL
    }
}