use std::path::PathBuf; use std::{env, fs}; use rusqlite::Connection; pub const DBNAME: &str = "cats.db"; pub const TABLE_CAT_COLOR: &str = "cat_colors"; pub const TABLE_CAT: &str = "cats"; pub fn connect() -> anyhow::Result { let mut wd = env::current_dir()?; let mut dbpath: Option = None; // does the current directory have a data/cats.db ? (the file need not exist, we can create it // if data exists) { let mut wddata = wd.clone(); wddata.push("data"); if wddata.exists() { println!("found {DBNAME} in {:?}", &wddata); wddata.push(DBNAME); dbpath = Some(wddata); } } // otherwise, does the current or any higher directories contain a cats.db ? 'search_dir: while dbpath.is_none() { for entry in fs::read_dir(&wd)? { let entry = entry?; if entry.file_name() == DBNAME { println!("found {DBNAME} in {:?}", wd); dbpath = Some(entry.path()); break 'search_dir; } } if !wd.pop() { // we are at the root! break 'search_dir; } } // if all fails, use $PWD/data/cats.db if dbpath.is_none() { println!( "No {DBNAME} found, using {:?}/data/{DBNAME}", env::current_dir()? ); fs::create_dir("data")?; let mut path = env::current_dir()?; path.push("data"); path.push(DBNAME); dbpath = Some(path); } let conn = Connection::open(dbpath.unwrap())?; Ok(conn) } pub fn setup(conn: &Connection) -> anyhow::Result<()> { conn.execute( &format!( "CREATE TABLE IF NOT EXISTS {TABLE_CAT_COLOR} ( id INTEGER PRIMARY KEY, name TEXT NOT NULL UNIQUE COLLATE NOCASE )" ), (), )?; conn.execute( &format!( "CREATE TABLE IF NOT EXISTS {TABLE_CAT} ( id INTEGER PRIMARY KEY, name TEXT NOT NULL COLLATE NOCASE, color_id INTEGER NOT NULL REFERENCES CAT_COLORS(ID) )" ), (), )?; Ok(()) } // color names are unique, so we use the color name instead of the id because it's easier pub fn check_if_color_exists(conn: &Connection, color: &str) -> anyhow::Result { let mut stmt = conn.prepare(&format!( "SELECT EXISTS( SELECT 1 FROM {TABLE_CAT_COLOR} WHERE name = (LOWER(?1)))" ))?; // we need the return or the temporary result is dropped somehow #[allow(clippy::needless_return)] return Ok(stmt .query([color.to_string()])? .next()? .unwrap() .get::(0)? == 1); } // note that cat names are not UNIQUE, so we need to use the id instead pub fn check_if_cat_exists(conn: &Connection, id: usize) -> anyhow::Result { let mut stmt = conn.prepare(&format!( "SELECT EXISTS( SELECT 1 FROM {TABLE_CAT} WHERE id = (?1))" ))?; // we need the return or the temporary result is dropped somehow #[allow(clippy::needless_return)] return Ok(stmt.query([id])?.next()?.unwrap().get::(0)? == 1); } /// Add a new color to the cat_colors table /// /// Will fail if the sql fails /// /// If the color already exists, will return Ok(id) /// /// The table stores the name in lowercase /// /// Returns the id of the color pub fn new_color(conn: &Connection, color: &str) -> anyhow::Result { if check_if_color_exists(conn, color)? { return get_color_id(conn, color); } conn.execute( &format!("INSERT INTO {TABLE_CAT_COLOR} (name) VALUES (LOWER(?1))"), [color.to_string()], )?; get_color_id(conn, color) } pub fn get_color_id(conn: &Connection, color: &str) -> anyhow::Result { Ok(conn.query_row( &format!("SELECT id FROM {TABLE_CAT_COLOR} WHERE name = (LOWER(?1))"), [color.to_string()], |row| row.get::(0), )?) } pub fn get_color_by_id(conn: &Connection, id: usize) -> anyhow::Result> { let maybe = conn.query_row( &format!("SELECT name FROM {TABLE_CAT_COLOR} WHERE id = (?1)"), [id.to_string()], |row| row.get::(0), ); if let Result::Ok(color) = maybe { Ok(Some(color)) } else { let err: rusqlite::Error = maybe.unwrap_err(); if matches!(err, rusqlite::Error::QueryReturnedNoRows) { Ok(None) } else { Err(err.into()) } } } pub fn new_cat(conn: &Connection, name: &str, color_id: usize) -> anyhow::Result { conn.execute( &format!("INSERT INTO {TABLE_CAT} (name, color_id) VALUES (LOWER(?1), ?2)"), [name.to_string(), color_id.to_string()], )?; Ok(conn.last_insert_rowid() as usize) } pub fn get_cat_color(conn: &Connection, id: usize) -> anyhow::Result { Ok(conn.query_row( &format!("SELECT cc.name FROM {TABLE_CAT_COLOR} cc, {TABLE_CAT} c WHERE c.id = (?1) AND c.color_id = cc.id"), [id.to_string()], |row| row.get::(0), )?) }