use std::io::{Error, ErrorKind};

use rusqlite::Connection;

use crate::{
    config::Cfg,
    data::{list::List, modloader::Modloader},
    error::{EType, MLErr, MLE},
};

//MODS
/// # Errors
pub fn mods_insert(config: &Cfg, id: &str, slug: &str, name: &str) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        "INSERT INTO mods (id, slug, title) VALUES (?1, ?2, ?3)",
        [id, slug, name.replace('\'', "").as_str()],
    )?;

    Ok(())
}

/// # Errors
pub fn mods_get_all_ids(
    config: &Cfg,
) -> Result<Vec<String>, Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut mods: Vec<String> = Vec::new();

    let mut stmt = connection.prepare("SELECT id FROM mods")?;
    let id_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for id in id_iter {
        mods.push(id?);
    }

    if mods.is_empty() {
        Err(Box::new(Error::new(ErrorKind::NotFound, "NO_MODS_ALL")))
    } else {
        Ok(mods)
    }
}

/// Get mod id based on the slug or name
/// # Arguments
///
///* `data` - file directory of the database
///* `slug` - Slug or Name of a mod
///
/// # Errors
///
/// Will return `MLError` when no mod id is found
pub fn mods_get_id(data: &str, slug: &str) -> MLE<String> {
    let data = format!("{data}/data.db");
    let connection = Connection::open(data)?;

    let mut mod_id = String::new();

    //get from slug
    let mut stmt = connection.prepare("SELECT id FROM mods WHERE slug = ?")?;
    let id_iter = stmt.query_map([slug], |row| row.get::<usize, String>(0))?;

    for id in id_iter {
        mod_id = id?;
    }
    //get from id if no slug found
    if mod_id.is_empty() {
        let mut stmt =
            connection.prepare("SELECT id FROM mods WHERE id = ?")?;
        let id_iter =
            stmt.query_map([slug], |row| row.get::<usize, String>(0))?;

        for id in id_iter {
            mod_id = id?;
        }
    }
    //get from title if no id found from slug
    if mod_id.is_empty() {
        let mut stmt =
            connection.prepare("SELECT id FROM mods WHERE title = ?")?;
        let id_iter =
            stmt.query_map([slug], |row| row.get::<usize, String>(0))?;

        for id in id_iter {
            mod_id = id?;
        }
    }

    if mod_id.is_empty() {
        return Err(MLErr::new(EType::DBError, "GI_MOD_NOT_FOUND"));
    };

    Ok(mod_id)
}

pub struct ModInfo {
    pub slug: String,
    pub title: String,
}

/// # Errors
pub fn mods_get_info(config: &Cfg, id: &str) -> MLE<ModInfo> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut mod_info: Option<ModInfo> = None;
    let mut stmt =
        connection.prepare("SELECT title, slug FROM mods WHERE id = ?")?;
    let name_iter = stmt.query_map([id], |row| {
        Ok(vec![
            row.get::<usize, String>(0)?,
            row.get::<usize, String>(1)?,
        ])
    })?;

    for info in name_iter {
        let i = info?;
        mod_info = Some(ModInfo {
            title: String::from(&i[0]),
            slug: String::from(&i[1]),
        });
    }

    if mod_info.is_none() {
        Err(MLErr::new(EType::DBError, "GN_MOD_NOT_FOUND"))
    } else {
        Ok(mod_info.ok_or(MLErr::new(EType::Other, "mod_info"))?)
    }
}

/// # Errors
pub fn mods_remove(config: &Cfg, id: &str) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute("DELETE FROM mods WHERE id = ?", [id])?;

    Ok(())
}

#[derive(Debug, Clone, PartialEq, Eq)]
pub struct DBModlistVersions {
    pub mod_id: String,
    pub versions: String,
}

/// # Errors
pub fn mods_get_versions(
    config: &Cfg,
    mods: &[String],
) -> MLE<Vec<DBModlistVersions>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    if mods.is_empty() {
        return Err(MLErr::new(EType::ArgumentError, "MODS_NO_INPUT"));
    }

    let mut wherestr = String::from("WHERE");
    for (i, id) in mods.iter().enumerate() {
        let mut or = " OR";
        if i == mods.len() - 1 {
            or = "";
        };
        wherestr = format!("{wherestr} id = '{id}'{or}");
    }

    let mut versionmaps: Vec<DBModlistVersions> = Vec::new();
    let mut stmt = connection.prepare(
        format!("SELECT id, versions, title FROM mods {wherestr}").as_str(),
    )?;
    let id_iter = stmt.query_map([], |row| {
        Ok(vec![
            row.get::<usize, String>(0)?,
            row.get::<usize, String>(1)?,
            row.get::<usize, String>(2)?,
        ])
    })?;

    for ver in id_iter {
        let version = ver?;
        versionmaps.push(DBModlistVersions {
            mod_id: String::from(&version[0]),
            versions: String::from(&version[1]),
        });
    }

    if versionmaps.is_empty() {
        Err(MLErr::new(EType::DBError, "MODS_MODS_NOT_FOUND"))
    } else {
        Ok(versionmaps)
    }
}

//userlist
/// # Errors
pub fn userlist_insert(
    config: &Cfg,
    list_id: &str,
    mod_id: &str,
    current_version: &str,
    applicable_versions: &[String],
    current_link: &str,
    set_version: bool,
) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let sv = if set_version { "1" } else { "0" };

    connection.execute(
        format!("INSERT INTO {list_id} VALUES (?1, ?2, ?3, ?4, 'NONE', ?5)")
            .as_str(),
        [
            mod_id,
            current_version,
            applicable_versions.join("|").as_str(),
            current_link,
            sv,
        ],
    )?;

    Ok(())
}

/// # Errors
pub fn userlist_get_all_ids(config: &Cfg, list_id: &str) -> MLE<Vec<String>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut mod_ids: Vec<String> = Vec::new();
    let mut stmt =
        connection.prepare(format!("SELECT mod_id FROM {list_id}").as_str())?;
    let id_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for id in id_iter {
        mod_ids.push(id?);
    }

    if mod_ids.is_empty() {
        Err(MLErr::new(
            EType::DBError,
            &format!("NO_MODS_USERLIST{list_id}"),
        ))
    } else {
        Ok(mod_ids)
    }
}

/// # Errors
pub fn userlist_remove(config: &Cfg, list_id: &str, mod_id: &str) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        format!("DELETE FROM {list_id} WHERE mod_id = ?").as_str(),
        [mod_id],
    )?;
    Ok(())
}

/// # Errors
pub fn userlist_get_applicable_versions(
    config: &Cfg,
    list_id: &str,
    mod_id: String,
) -> MLE<String> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut version: String = String::new();
    let mut stmt = connection.prepare(
        format!("SELECT applicable_versions FROM {list_id} WHERE mod_id = ?")
            .as_str(),
    )?;
    let ver_iter =
        stmt.query_map([mod_id], |row| row.get::<usize, String>(0))?;

    for ver in ver_iter {
        version = ver?;
    }

    if version.is_empty() {
        Err(MLErr::new(EType::DBError, "GAV_MOD_NOT_FOUND"))
    } else {
        Ok(version)
    }
}

/// # Errors
pub fn userlist_get_all_applicable_versions_with_mods(
    config: &Cfg,
    list_id: &str,
) -> MLE<Vec<(String, String)>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut versions: Vec<(String, String)> = Vec::new();
    let mut stmt = connection.prepare(
        format!("SELECT mod_id, applicable_versions FROM {list_id}").as_str(),
    )?;
    let id_iter = stmt.query_map([], |row| {
        Ok(vec![
            row.get::<usize, String>(0)?,
            row.get::<usize, String>(1)?,
        ])
    })?;

    for ver in id_iter {
        let out = ver?;
        versions.push((out[0].clone(), out[1].clone()));
    }

    if versions.is_empty() {
        return Err(MLErr::new(EType::DBError, "NO_MODS_ON_LIST"));
    };

    Ok(versions)
}

/// # Errors
pub fn userlist_get_current_version(
    config: &Cfg,
    list_id: &str,
    mod_id: &str,
) -> MLE<String> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut version: String = String::new();
    let mut stmt = connection.prepare(
        format!("SELECT current_version FROM {list_id} WHERE mod_id = ?")
            .as_str(),
    )?;
    let ver_iter =
        stmt.query_map([&mod_id], |row| row.get::<usize, String>(0))?;

    for ver in ver_iter {
        version = ver?;
    }

    if version.is_empty() {
        Err(MLErr::new(EType::DBError, "GCV_MOD_NOT_FOUND"))
    } else {
        Ok(version)
    }
}

/// # Errors
pub fn userlist_get_all_current_version_ids(
    config: &Cfg,
    list_id: &str,
) -> MLE<Vec<String>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut versions: Vec<String> = Vec::new();
    let mut stmt = connection
        .prepare(format!("SELECT current_version FROM {list_id}").as_str())?;
    let id_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for id in id_iter {
        versions.push(id?);
    }

    if versions.is_empty() {
        return Err(MLErr::new(EType::DBError, "NO_MODS_ON_LIST"));
    };

    Ok(versions)
}

/// # Errors
pub fn userlist_get_all_current_versions_with_mods(
    config: &Cfg,
    list_id: &str,
) -> Result<Vec<(String, String)>, Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut versions: Vec<(String, String)> = Vec::new();
    let mut stmt = connection.prepare(
        format!("SELECT mod_id, current_version FROM {list_id}").as_str(),
    )?;
    let id_iter = stmt.query_map([], |row| {
        Ok(vec![
            row.get::<usize, String>(0)?,
            row.get::<usize, String>(1)?,
        ])
    })?;

    for ver in id_iter {
        let out = ver?;
        versions.push((out[0].clone(), out[1].clone()));
    }

    if versions.is_empty() {
        return Err(Box::new(std::io::Error::new(
            ErrorKind::Other,
            "NO_MODS_ON_LIST",
        )));
    };

    Ok(versions)
}

/// # Errors
pub fn userlist_get_set_version(
    config: &Cfg,
    list_id: &str,
    mod_id: &str,
) -> MLE<bool> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut set_version: bool = false;
    let mut stmt = connection.prepare(
        format!("SELECT set_version FROM {list_id} WHERE mod_id = ?").as_str(),
    )?;
    let ver_iter =
        stmt.query_map([&mod_id], |row| row.get::<usize, bool>(0))?;

    for ver in ver_iter {
        set_version = ver?;
    }

    Ok(set_version)
}

/// # Errors
pub fn userlist_change_versions(
    config: &Cfg,
    list_id: &str,
    current_version: String,
    versions: String,
    link: String,
    mod_id: String,
) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(format!("UPDATE {list_id} SET current_version = ?1, applicable_versions = ?2, current_download = ?3 WHERE mod_id = ?4").as_str(), [current_version, versions, link, mod_id])?;
    Ok(())
}

/// # Errors
pub fn userlist_add_disabled_versions(
    config: &Cfg,
    list_id: &str,
    disabled_version: String,
    mod_id: String,
) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let currently_disabled_versions =
        userlist_get_disabled_versions(config, list_id, String::from(&mod_id))?;
    let disabled_versions = if currently_disabled_versions == "NONE" {
        disabled_version
    } else {
        format!("{currently_disabled_versions}|{disabled_version}")
    };

    connection.execute(
        format!(
            "UPDATE {list_id} SET disabled_versions = ?1 WHERE mod_id = ?2"
        )
        .as_str(),
        [disabled_versions, mod_id],
    )?;
    Ok(())
}

/// # Errors
pub fn userlist_get_disabled_versions(
    config: &Cfg,
    list_id: &str,
    mod_id: String,
) -> MLE<String> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut version: String = String::new();
    let mut stmt = connection.prepare(
        format!("SELECT disabled_versions FROM {list_id} WHERE mod_id = ?")
            .as_str(),
    )?;
    let ver_iter =
        stmt.query_map([mod_id], |row| row.get::<usize, String>(0))?;

    for ver in ver_iter {
        version = ver?;
    }

    if version.is_empty() {
        Err(MLErr::new(EType::DBError, "GDV_MOD_NOT_FOUND"))
    } else {
        Ok(version)
    }
}

/// # Errors
pub fn userlist_get_all_downloads(
    config: &Cfg,
    list_id: &str,
) -> Result<Vec<String>, Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut links: Vec<String> = Vec::new();
    let mut stmt = connection
        .prepare(format!("SELECT current_download FROM {list_id}").as_str())?;
    let link_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for link in link_iter {
        let l = link?;
        links.push(l);
    }

    if links.is_empty() {
        return Err(Box::new(std::io::Error::new(
            ErrorKind::Other,
            "NO_MODS_ON_LIST",
        )));
    };

    Ok(links)
}

//lists
///Inserts into lists table and creates new table
/// # Errors
pub fn lists_insert(
    config: &Cfg,
    id: &str,
    mc_version: &str,
    mod_loader: &Modloader,
    download_folder: &str,
) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        "INSERT INTO lists VALUES (?1, ?2, ?3, ?4)",
        [id, mc_version, &mod_loader.to_string(), download_folder],
    )?;
    connection.execute(format!("CREATE TABLE {id}( 'mod_id' TEXT, 'current_version' TEXT, 'applicable_versions' BLOB, 'current_download' TEXT, 'disabled_versions' TEXT DEFAULT 'NONE', 'set_version' INTEGER, CONSTRAINT {id}_PK PRIMARY KEY (mod_id) )").as_str(), [])?;

    Ok(())
}

/// # Errors
pub fn lists_remove(config: &Cfg, id: &str) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute("DELETE FROM lists WHERE id = ?", [&id])?;
    connection.execute(format!("DROP TABLE {id}").as_str(), [])?;
    Ok(())
}

/// # Errors
pub fn lists_get(config: &Cfg, list_id: &str) -> MLE<List> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut list = List {
        id: String::new(),
        mc_version: String::new(),
        modloader: Modloader::Fabric,
        download_folder: String::new(),
    };
    let mut stmt = connection.prepare(
        "SELECT mc_version, modloader, download_folder FROM lists WHERE id = ?",
    )?;

    let list_iter = stmt.query_map([&list_id], |row| {
        Ok(vec![
            row.get::<usize, String>(0)?,
            row.get::<usize, String>(1)?,
            row.get::<usize, String>(2)?,
        ])
    })?;

    for l in list_iter {
        let li = l?;
        list = List {
            id: list_id.to_string(),
            mc_version: String::from(&li[0]),
            modloader: Modloader::from(&li[1])?,
            download_folder: String::from(&li[2]),
        };
    }

    if list.id.is_empty() {
        return Err(MLErr::new(EType::DBError, "LIST_NOT_FOUND"));
    }

    Ok(list)
}

/// # Errors
pub fn lists_version(config: &Cfg, list_id: &str, version: &str) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        "UPDATE lists SET mc_version = ? WHERE id = ?",
        [version, list_id],
    )?;
    Ok(())
}

/// # Errors
pub fn lists_get_all_ids(config: &Cfg) -> MLE<Vec<String>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut list_ids: Vec<String> = Vec::new();
    let mut stmt = connection.prepare("SELECT id FROM lists")?;
    let id_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for id in id_iter {
        list_ids.push(id?);
    }

    if list_ids.is_empty() {
        Err(MLErr::new(EType::DBError, "NO_LISTS"))
    } else {
        Ok(list_ids)
    }
}

//config
/// # Errors
pub fn config_change_current_list(config: &Cfg, id: &str) -> MLE<()> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        "UPDATE user_config SET value = ? WHERE id = 'current_list'",
        [id],
    )?;
    Ok(())
}

/// # Errors
pub fn config_get_current_list(config: &Cfg) -> MLE<String> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut list_id = String::new();
    let mut stmt = connection
        .prepare("SELECT value FROM user_config WHERE id = 'current_list'")?;
    let list_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for list in list_iter {
        list_id = list?;
    }

    if list_id.is_empty() {
        return Err(MLErr::new(EType::DBError, "NO_CURRENT_LIST"));
    }

    Ok(list_id)
}

//SETUP(UPDATES)
/// # Errors
pub fn s_userlist_update_download(
    config: &Cfg,
    list_id: &str,
    mod_id: String,
    link: String,
) -> Result<(), Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        format!("UPDATE {list_id} SET current_download = ?1 WHERE mod_id = ?2")
            .as_str(),
        [link, mod_id],
    )?;
    Ok(())
}

/// # Errors
pub fn s_config_create_version(
    config: &Cfg,
) -> Result<(), Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        "INSERT INTO 'user_config' VALUES ( 'db_version', '0.2' )",
        (),
    )?;
    Ok(())
}

/// # Errors
pub fn s_config_update_version(
    config: &Cfg,
    ver: String,
) -> Result<(), Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    connection.execute(
        "UPDATE user_config SET value = ? WHERE id = 'db_version'",
        [ver],
    )?;
    Ok(())
}

/// # Errors
pub fn s_config_get_version(
    config: &Cfg,
) -> Result<String, Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut version: String = String::new();
    let mut stmt = connection
        .prepare("SELECT value FROM user_config WHERE id = 'db_version'")?;
    let ver_iter = stmt.query_map([], |row| row.get::<usize, String>(0))?;

    for ver in ver_iter {
        version = ver?;
    }

    if version.is_empty() {
        return Err(Box::new(std::io::Error::new(
            ErrorKind::Other,
            "NO_DBVERSION",
        )));
    };
    Ok(version)
}

/// # Errors
pub fn s_insert_column(
    config: &Cfg,
    table: &str,
    column: &str,
    c_type: &str,
    default: Option<String>,
) -> Result<(), Box<dyn std::error::Error>> {
    let data = format!("{}/data.db", config.data);
    let connection = Connection::open(data)?;

    let mut sql = format!("ALTER TABLE {table} ADD '{column}' {c_type}");

    if default.is_some() {
        sql = format!(
            "{} DEFAULT {}",
            sql,
            default.ok_or(MLErr::new(EType::Other, "errornous default"))?
        );
    }

    connection.execute(sql.as_str(), ())?;
    Ok(())
}

/// # Errors
pub fn setup(path: &str) -> MLE<()> {
    let connection = Connection::open(path)?;

    connection.execute_batch(
        "CREATE TABLE 'user_config' ( 'id' TEXT, 'value' TEXT );
         CREATE TABLE 'mods' ( 'id' TEXT, 'slug' TEXT, 'title' TEXT, CONSTRAINT mods_PK PRIMARY KEY (id) );
         CREATE TABLE 'lists' ( 'id' TEXT, 'mc_version' TEXT, 'modloader' TEXT, 'download_folder' TEXT );
         INSERT INTO 'user_config' VALUES ( 'db_version', '0.5' );
         INSERT INTO 'user_config' VALUES ( 'current_list', '...' )",
    )?;

    Ok(())
}