rs-basic/crates/sqlite-demo/src/db.rs
Christoph J. Scherr ce71b2ceec
Some checks failed
cargo devel CI / cargo CI (push) Failing after 32s
reorganize the workspace
2025-03-14 22:24:47 +01:00

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),
)?)
}