rustbasic-cli 0.0.11

CLI tool for the RustBasic Framework.
Documentation
use std::fs;
use rustbasic_core::Config;
use rustbasic_core::database::connect;
use base64::{Engine as _, engine::general_purpose};
use rand::Rng;
use regex::Regex;
use colored::*;
use sea_orm::ConnectionTrait;

pub async fn clear_cache() {
    println!("\n{}", "๐Ÿงน Cleaning Cache & Logs...".magenta().bold());

    // 1. Clear Logs
    let log_dir = "storage/logs";
    if let Ok(entries) = fs::read_dir(log_dir) {
        let mut count = 0;
        for entry in entries.flatten() {
            let path = entry.path();
            if path.is_file() {
                let _ = fs::OpenOptions::new()
                    .write(true)
                    .truncate(true)
                    .open(&path);
                count += 1;
            }
        }
        println!("   {} Folder storage/logs telah dikosongkan. ({} file dibersihkan)", "โœ… Logs:".green(), count);
    } else {
        println!("   {} Folder storage/logs tidak ditemukan.", "โš ๏ธ  Logs:".yellow());
    }

    // 2. Clear Sessions in DB
    let cfg = Config::load();
    let db = connect(&cfg).await;
    
    let truncate_sql = if cfg.db_connection == "mysql" {
        "TRUNCATE TABLE sessions"
    } else {
        "DELETE FROM sessions"
    };

    match db.execute(sea_orm::Statement::from_string(cfg.db_backend(), truncate_sql.to_string())).await {
        Ok(_) => println!("   {} Tabel sessions telah dikosongkan.", "โœ… Sessions:".green()),
        Err(e) => println!("   {} Gagal membersihkan tabel sessions. ({})", "โŒ Error:".red(), e),
    }

    println!("\n{}", "โœจ Cache berhasil dibersihkan!".green().bold());
}

pub fn generate_app_key() {
    println!("\n{}", "๐Ÿ”‘ Generating Application Key...".magenta().bold());

    let mut key = [0u8; 32];
    rand::rng().fill_bytes(&mut key);
    
    let encoded = general_purpose::STANDARD.encode(key);
    let key_str = format!("base64:{}", encoded);
    
    let env_path = ".env";
    match fs::read_to_string(env_path) {
        Ok(content) => {
            let re = Regex::new(r"(?m)^APP_KEY=.*").unwrap();
            let new_content = if re.is_match(&content) {
                re.replace(&content, &format!("APP_KEY={}", key_str)).to_string()
            } else {
                format!("{}\nAPP_KEY={}", content.trim_end(), key_str)
            };

            if let Err(e) = fs::write(env_path, new_content) {
                println!("{} Gagal menulis ke file .env: {}", "โŒ Error:".red(), e);
            } else {
                println!("{} {}", "โœ… Application key set successfully:".green(), key_str.cyan());
                println!("{}", "๐Ÿ’ก Pastikan untuk tidak membagikan APP_KEY ini ke publik!".dimmed());
            }
        }
        Err(_) => {
            println!("{} File .env tidak ditemukan.", "โŒ Error:".red());
        }
    }
}

pub async fn ensure_session() {
    if !std::path::Path::new(".env").exists() {
        return;
    }

    let _ = dotenvy::dotenv();
    let cfg = Config::load();

    // 1. Hubungkan ke database dengan aman (tanpa panik jika gagal)
    let db_url = if cfg.db_connection == "mysql" {
        format!(
            "mysql://{}:{}@{}:{}/{}",
            cfg.db_username, cfg.db_password, cfg.db_host, cfg.db_port, cfg.db_database
        )
    } else {
        format!("sqlite:database/{}.sqlite?mode=rwc", cfg.db_database)
    };

    // Pastikan folder database ada untuk sqlite
    if cfg.db_connection != "mysql" {
        let _ = std::fs::create_dir_all("database");
    }

    let mut opt = sea_orm::ConnectOptions::new(db_url);
    opt.max_connections(5)
       .connect_timeout(std::time::Duration::from_secs(3))
       .sqlx_logging(false);

    let db = match sea_orm::Database::connect(opt).await {
        Ok(conn) => conn,
        Err(_) => {
            // Abaikan jika database belum siap/tidak bisa terhubung (agar perintah seperti migrate/help/new tidak crash)
            return;
        }
    };

    // 2. Pastikan tabel sessions ada di database
    let create_table_sql = if cfg.db_connection == "mysql" {
        "CREATE TABLE IF NOT EXISTS sessions (
            id VARCHAR(255) PRIMARY KEY,
            payload TEXT NOT NULL,
            last_activity BIGINT NOT NULL,
            ip_address VARCHAR(45)
        )"
    } else {
        "CREATE TABLE IF NOT EXISTS sessions (
            id TEXT PRIMARY KEY,
            payload TEXT NOT NULL,
            last_activity BIGINT NOT NULL,
            ip_address TEXT
        )"
    };

    let stmt = sea_orm::Statement::from_string(db.get_database_backend(), create_table_sql.to_string());
    if db.execute(stmt).await.is_err() {
        return;
    }

    // 3. Baca dan verifikasi session ID
    let session_file = ".rustbasic_session";
    let mut session_id = String::new();
    let mut need_to_write = false;

    if let Ok(content) = fs::read_to_string(session_file) {
        session_id = content.trim().to_string();
    }

    let mut session_exists = false;

    if !session_id.is_empty() {
        let check_sql = format!("SELECT id FROM sessions WHERE id = '{}'", session_id);
        let stmt = sea_orm::Statement::from_string(db.get_database_backend(), check_sql);
        if let Ok(Some(_)) = db.query_one(stmt).await {
            session_exists = true;
        }
    }

    if !session_exists {
        let mut bytes = [0u8; 32];
        rand::rng().fill_bytes(&mut bytes);
        session_id = general_purpose::STANDARD.encode(bytes);
        need_to_write = true;

        let now = chrono::Utc::now().timestamp();
        let insert_sql = format!(
            "INSERT INTO sessions (id, payload, last_activity, ip_address) VALUES ('{}', '{}', {}, '{}')",
            session_id, "{}", now, "127.0.0.1"
        );
        let stmt = sea_orm::Statement::from_string(db.get_database_backend(), insert_sql);
        if db.execute(stmt).await.is_ok() {
            println!("โœจ {} ({})", "Session baru berhasil dibuat dan disimpan di database.".green().bold(), session_id.cyan());
        }
    } else {
        let now = chrono::Utc::now().timestamp();
        let update_sql = format!(
            "UPDATE sessions SET last_activity = {} WHERE id = '{}'",
            now, session_id
        );
        let stmt = sea_orm::Statement::from_string(db.get_database_backend(), update_sql);
        let _ = db.execute(stmt).await;
    }

    if need_to_write {
        let _ = fs::write(session_file, &session_id);
    }
}