Compware/src/db.rs

289 lines
11 KiB
Rust
Raw Normal View History

2025-01-20 18:49:54 +03:00
#[cfg(feature = "ssr")]
mod db_impl {
use rusqlite::{Connection, Error};
use serde::{Deserialize, Serialize};
use std::sync::Arc;
use tokio::sync::Mutex;
use leptos::logging;
use serde_json;
use std::collections::HashMap;
2025-01-20 18:49:54 +03:00
// Define a struct to represent a database connection
#[derive(Debug)]
pub struct Database {
conn: Arc<Mutex<Connection>>,
}
2025-01-20 18:49:54 +03:00
impl Database {
// Create a new database connection
pub fn new(db_path: &str) -> Result<Self, Error> {
let conn = Connection::open(db_path)?;
logging::log!("Database connection established at: {}", db_path);
Ok(Database {
conn: Arc::new(Mutex::new(conn)),
})
2025-01-20 18:49:54 +03:00
}
2025-01-20 18:49:54 +03:00
// Create the database schema
pub async fn create_schema(&self) -> Result<(), Error> {
let conn = self.conn.lock().await;
// 1. Properties table
conn.execute_batch(
"CREATE TABLE IF NOT EXISTS properties (
id INTEGER PRIMARY KEY,
name TEXT NOT NULL UNIQUE,
global_usage_count INTEGER DEFAULT 0
);"
)?;
// 2. URLs table
conn.execute_batch(
"CREATE TABLE IF NOT EXISTS urls (
id INTEGER PRIMARY KEY,
url TEXT NOT NULL UNIQUE, // Enforce unique URLs
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);",
)?;
logging::log!("URLs table created or verified");
// 3. Items table
conn.execute_batch(
"CREATE TABLE IF NOT EXISTS items (
2025-01-20 18:49:54 +03:00
id TEXT PRIMARY KEY,
url_id INTEGER NOT NULL,
2025-01-20 18:49:54 +03:00
name TEXT NOT NULL,
description TEXT,
wikidata_id TEXT,
FOREIGN KEY (url_id) REFERENCES urls(id) ON DELETE CASCADE
);",
)?;
logging::log!("Items table updated with foreign key to URLs table");
// 4. Junction table for custom properties
conn.execute_batch(
"CREATE TABLE IF NOT EXISTS item_properties (
item_id TEXT NOT NULL,
property_id INTEGER NOT NULL,
value TEXT NOT NULL,
PRIMARY KEY (item_id, property_id),
FOREIGN KEY (item_id) REFERENCES items(id) ON DELETE CASCADE,
FOREIGN KEY (property_id) REFERENCES properties(id) ON DELETE CASCADE
);"
)?;
2025-01-20 18:49:54 +03:00
Ok(())
}
// Insert a new URL into the database
pub async fn insert_url(&self, url: &str) -> Result<i64, Error> {
let conn = self.conn.lock().await;
let mut stmt = conn.prepare("INSERT INTO urls (url) VALUES (?)")?;
let url_id = stmt.insert(&[url])?;
logging::log!("URL inserted: {}", url);
Ok(url_id)
}
2025-01-20 18:49:54 +03:00
// Insert a new item into the database
pub async fn insert_item(&self, url_id: i64, item: &DbItem) -> Result<(), Error> {
let conn = self.conn.lock().await;
2025-01-20 18:49:54 +03:00
let wikidata_id = item.wikidata_id.as_ref().map(|s| s.as_str()).unwrap_or("");
conn.execute(
"INSERT INTO items (id, name, description, wikidata_id, custom_properties, url_id)
VALUES (?, ?, ?, ?, ?, ?)
ON CONFLICT(id) DO UPDATE SET
name = excluded.name,
description = excluded.description,
wikidata_id = excluded.wikidata_id,
custom_properties = excluded.custom_properties;",
&[
&item.id,
&item.name,
&item.description,
&wikidata_id.to_string(),
&item.custom_properties,
&url_id.to_string(),
],
)?;
logging::log!("Item inserted: {}", item.id);
2025-01-20 18:49:54 +03:00
Ok(())
}
pub async fn delete_item(&self, item_id: &str) -> Result<(), Error> {
let conn = self.conn.lock().await;
conn.execute("DELETE FROM items WHERE id = ?", &[item_id])?;
logging::log!("Item deleted: {}", item_id);
Ok(())
}
pub async fn delete_property(&self, property: &str) -> Result<(), Error> {
let conn = self.conn.lock().await;
let query = format!("UPDATE items SET custom_properties = json_remove(custom_properties, '$.{}')", property);
conn.execute(&query, []).map_err(|e| Error::from(e))?;
logging::log!("Property deleted: {}", property);
Ok(())
}
2025-01-20 18:49:54 +03:00
// Retrieve all items from the database
pub async fn get_items(&self) -> Result<Vec<DbItem>, Error> {
let conn = self.conn.lock().await;
let mut stmt = conn.prepare("SELECT * FROM items;")?;
2025-01-20 18:49:54 +03:00
let items = stmt.query_map([], |row| {
Ok(DbItem {
id: row.get(0)?,
name: row.get(1)?,
description: row.get(2)?,
wikidata_id: row.get(3)?,
custom_properties: row.get(4)?,
})
})?;
let mut result = Vec::new();
for item in items {
result.push(item?);
}
logging::log!("Fetched {} items from the database", result.len()); // Log with Leptos
2025-01-20 18:49:54 +03:00
Ok(result)
}
// Retrieve all items from the database for a specific URL
pub async fn get_items_by_url(&self, url: &str) -> Result<Vec<DbItem>, Error> {
let conn = self.conn.lock().await;
let url_id: i64 = conn.query_row("SELECT id FROM urls WHERE url = ?", &[url], |row| row.get(0))?;
let mut stmt = conn.prepare("SELECT * FROM items WHERE url_id = ?")?;
let items = stmt.query_map(&[&url_id], |row| {
Ok(DbItem {
id: row.get(0)?,
name: row.get(1)?,
description: row.get(2)?,
wikidata_id: row.get(3)?,
custom_properties: row.get(4)?,
})
})?;
let mut result = Vec::new();
for item in items {
result.push(item?);
}
logging::log!("Fetched {} items from the database for URL: {}", result.len(), url);
Ok(result)
}
async fn get_url_id(&self, url: &str) -> Result<Option<i64>, Error> {
let conn = self.conn.lock().await;
conn.query_row(
"SELECT id FROM urls WHERE url = ?",
&[url],
|row| row.get(0)
)
}
async fn get_or_create_property(&self, prop: &str) -> Result<i64, Error> {
let conn = self.conn.lock().await;
// Check existing
let exists: Result<i64, _> = conn.query_row(
"SELECT id FROM properties WHERE name = ?",
&[prop],
|row| row.get(0)
);
match exists {
Ok(id) => Ok(id),
Err(_) => {
conn.execute(
"INSERT INTO properties (name) VALUES (?)",
&[prop],
)?;
Ok(conn.last_insert_rowid())
}
}
}
// Insert a new item into the database for a specific URL
pub async fn insert_item_by_url(
&self,
url: &str,
item: &DbItem
) -> Result<(), Error> {
let conn = self.conn.lock().await;
// Get or create URL record
let url_id = match self.get_url_id(url).await {
Ok(Some(id)) => id,
_ => self.insert_url(url).await?,
};
// Insert item with URL relationship
conn.execute(
"INSERT INTO items (id, url_id, name, description, wikidata_id)
VALUES (?, ?, ?, ?, ?)",
&[&item.id, &url_id.to_string(), &item.name,
&item.description, &item.wikidata_id.as_ref().unwrap_or(&String::new())],
)?;
let custom_props: HashMap<String, String> = serde_json::from_str(&item.custom_properties)
.map_err(|e| Error::ToSqlConversionFailure(e.into()))?;
// Handle properties through junction table
for (prop, value) in custom_props {
let prop_id = self.get_or_create_property(&prop).await?;
conn.execute(
"INSERT INTO item_properties (item_id, property_id, value)
VALUES (?, ?, ?)",
&[&item.id, &prop_id.to_string(), &value],
)?;
}
Ok(())
}
// Delete an item from the database for a specific URL
pub async fn delete_item_by_url(&self, url: &str, item_id: &str) -> Result<(), Error> {
let conn = self.conn.lock().await;
let url_id: i64 = conn.query_row("SELECT id FROM urls WHERE url = ?", &[url], |row| row.get(0))?;
conn.execute("DELETE FROM items WHERE id = ? AND url_id = ?", &[item_id, &url_id.to_string()])?;
logging::log!("Item deleted from the database for URL: {}", url);
Ok(())
}
// Delete a property from the database for a specific URL
pub async fn delete_property_by_url(&self, url: &str, property: &str) -> Result<(), Error> {
let conn = self.conn.lock().await;
let url_id: i64 = conn.query_row("SELECT id FROM urls WHERE url = ?", &[url], |row| row.get(0))?;
let query = format!("UPDATE items SET custom_properties = json_remove(custom_properties, '$.{}') WHERE url_id = ?", property);
conn.execute(&query, &[&url_id.to_string()])?;
logging::log!("Property deleted from the database for URL: {}", url);
Ok(())
}
2025-01-20 18:49:54 +03:00
}
// Define a struct to represent an item in the database
#[derive(Debug, Deserialize, Serialize, Clone)]
2025-01-20 18:49:54 +03:00
pub struct DbItem {
pub id: String,
pub name: String,
pub description: String,
pub wikidata_id: Option<String>,
pub custom_properties: String,
}
// Implement conversion from DbItem to a JSON-friendly format
#[derive(Debug, Deserialize, Serialize, Clone)]
pub struct ItemResponse {
pub id: String,
pub name: String,
pub description: String,
pub wikidata_id: Option<String>,
pub custom_properties: String,
}
impl From<DbItem> for ItemResponse {
fn from(item: DbItem) -> Self {
ItemResponse {
id: item.id,
name: item.name,
description: item.description,
wikidata_id: item.wikidata_id,
custom_properties: item.custom_properties,
}
}
}
}
2025-01-20 18:49:54 +03:00
#[cfg(feature = "ssr")]
pub use db_impl::{Database, DbItem, ItemResponse};