Some checks failed
cargo devel CI / cargo CI (push) Failing after 32s
173 lines
5.1 KiB
Rust
173 lines
5.1 KiB
Rust
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<Connection> {
|
|
let mut wd = env::current_dir()?;
|
|
let mut dbpath: Option<PathBuf> = 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<bool> {
|
|
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::<usize, usize>(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<bool> {
|
|
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::<usize, usize>(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<usize> {
|
|
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<usize> {
|
|
Ok(conn.query_row(
|
|
&format!("SELECT id FROM {TABLE_CAT_COLOR} WHERE name = (LOWER(?1))"),
|
|
[color.to_string()],
|
|
|row| row.get::<usize, usize>(0),
|
|
)?)
|
|
}
|
|
|
|
pub fn get_color_by_id(conn: &Connection, id: usize) -> anyhow::Result<Option<String>> {
|
|
let maybe = conn.query_row(
|
|
&format!("SELECT name FROM {TABLE_CAT_COLOR} WHERE id = (?1)"),
|
|
[id.to_string()],
|
|
|row| row.get::<usize, String>(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<usize> {
|
|
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<String> {
|
|
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::<usize, String>(0),
|
|
)?)
|
|
}
|