fix(db): improve property handling
This commit is contained in:
parent
197e7be2a8
commit
0a05b41ffa
1 changed files with 67 additions and 50 deletions
115
src/db.rs
115
src/db.rs
|
@ -5,7 +5,7 @@ mod db_impl {
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use tokio::sync::Mutex;
|
use tokio::sync::Mutex;
|
||||||
use leptos::logging;
|
use leptos::logging;
|
||||||
use std::collections::HashMap;
|
use std::collections::{HashMap, HashSet};
|
||||||
use crate::models::item::Item;
|
use crate::models::item::Item;
|
||||||
use leptos::logging::log;
|
use leptos::logging::log;
|
||||||
|
|
||||||
|
@ -189,24 +189,22 @@ mod db_impl {
|
||||||
Ok(items.into_values().collect())
|
Ok(items.into_values().collect())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn get_or_create_property(&self, prop: &str) -> Result<i64, Error> {
|
async fn get_or_create_property(
|
||||||
let conn = self.conn.lock().await;
|
&self,
|
||||||
// Check existing
|
tx: &mut rusqlite::Transaction<'_>,
|
||||||
let exists: Result<i64, _> = conn.query_row(
|
prop: &str
|
||||||
|
) -> Result<i64, Error> {
|
||||||
|
match tx.query_row(
|
||||||
"SELECT id FROM properties WHERE name = ?",
|
"SELECT id FROM properties WHERE name = ?",
|
||||||
&[prop],
|
[prop],
|
||||||
|row| row.get(0)
|
|row| row.get::<_, i64>(0)
|
||||||
);
|
) {
|
||||||
|
|
||||||
match exists {
|
|
||||||
Ok(id) => Ok(id),
|
Ok(id) => Ok(id),
|
||||||
Err(_) => {
|
Err(rusqlite::Error::QueryReturnedNoRows) => {
|
||||||
conn.execute(
|
tx.execute("INSERT INTO properties (name) VALUES (?)", [prop])?;
|
||||||
"INSERT INTO properties (name) VALUES (?)",
|
Ok(tx.last_insert_rowid())
|
||||||
&[prop],
|
|
||||||
)?;
|
|
||||||
Ok(conn.last_insert_rowid())
|
|
||||||
}
|
}
|
||||||
|
Err(e) => Err(e.into()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -221,7 +219,7 @@ mod db_impl {
|
||||||
|
|
||||||
// 2. Transaction handling
|
// 2. Transaction handling
|
||||||
log!("[DB] Starting transaction");
|
log!("[DB] Starting transaction");
|
||||||
let tx = conn.transaction().map_err(|e| {
|
let mut tx = conn.transaction().map_err(|e| {
|
||||||
log!("[DB] Transaction start failed: {:?}", e);
|
log!("[DB] Transaction start failed: {:?}", e);
|
||||||
e
|
e
|
||||||
})?;
|
})?;
|
||||||
|
@ -267,41 +265,60 @@ mod db_impl {
|
||||||
)?;
|
)?;
|
||||||
log!("[DB] Item upserted successfully");
|
log!("[DB] Item upserted successfully");
|
||||||
// Property handling with enhanced logging
|
// Property handling with enhanced logging
|
||||||
log!("[DB] Processing {} properties", item.custom_properties.len());
|
log!("[DB] Synchronizing properties for item {}", item.id);
|
||||||
for (prop, value) in &item.custom_properties {
|
let existing_props = {
|
||||||
log!("[DB] Handling property: {}", prop);
|
// Prepare statement and collect existing properties
|
||||||
|
let mut stmt = tx.prepare(
|
||||||
// Property Lookup/Creation
|
"SELECT p.name, ip.value
|
||||||
let prop_id = match tx.query_row(
|
FROM item_properties ip
|
||||||
"SELECT id FROM properties WHERE name = ?",
|
JOIN properties p ON ip.property_id = p.id
|
||||||
[prop],
|
WHERE ip.item_id = ?"
|
||||||
|row| row.get::<_, i64>(0)
|
|
||||||
) {
|
|
||||||
Ok(id) => {
|
|
||||||
log!("[DB] Existing property ID: {}", id);
|
|
||||||
id
|
|
||||||
},
|
|
||||||
Err(rusqlite::Error::QueryReturnedNoRows) => {
|
|
||||||
log!("[DB] Creating new property: {}", prop);
|
|
||||||
tx.execute("INSERT INTO properties (name) VALUES (?)", [prop])?;
|
|
||||||
let id = tx.last_insert_rowid();
|
|
||||||
log!("[DB] New property ID: {}", id);
|
|
||||||
id
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
log!("[DB] Property lookup error: {:?}", e);
|
|
||||||
return Err(e.into());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
// Property Value Insertion
|
|
||||||
log!("[DB] Inserting property {} with value {}", prop, value);
|
|
||||||
tx.execute(
|
|
||||||
"INSERT INTO item_properties (item_id, property_id, value)
|
|
||||||
VALUES (?, ?, ?)",
|
|
||||||
rusqlite::params![&item.id, prop_id, &value],
|
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
|
let mapped_rows = stmt.query_map([&item.id], |row| {
|
||||||
|
Ok((row.get::<_, String>(0)?, row.get::<_, String>(1)?))
|
||||||
|
})?;
|
||||||
|
|
||||||
|
mapped_rows.collect::<Result<HashMap<String, String>, _>>()?
|
||||||
|
};
|
||||||
|
|
||||||
|
for (prop, value) in &item.custom_properties {
|
||||||
|
// Update existing or insert new
|
||||||
|
let prop_id = self.get_or_create_property(&mut tx, prop).await?;
|
||||||
|
if let Some(existing_value) = existing_props.get(prop) {
|
||||||
|
if existing_value != value {
|
||||||
|
log!("[DB] Updating property {} from '{}' to '{}'", prop, existing_value, value);
|
||||||
|
tx.execute(
|
||||||
|
"UPDATE item_properties
|
||||||
|
SET value = ?
|
||||||
|
WHERE item_id = ?
|
||||||
|
AND property_id = (SELECT id FROM properties WHERE name = ?)",
|
||||||
|
rusqlite::params![value, &item.id, prop],
|
||||||
|
)?;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log!("[DB] Adding new property {}", prop);
|
||||||
|
tx.execute(
|
||||||
|
"INSERT INTO item_properties (item_id, property_id, value)
|
||||||
|
VALUES (?, ?, ?)",
|
||||||
|
rusqlite::params![&item.id, prop_id, value],
|
||||||
|
)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Remove deleted properties
|
||||||
|
let current_props: HashSet<&str> = item.custom_properties.keys().map(|s| s.as_str()).collect();
|
||||||
|
for (existing_prop, _) in existing_props {
|
||||||
|
if !current_props.contains(existing_prop.as_str()) {
|
||||||
|
log!("[DB] Removing deleted property {}", existing_prop);
|
||||||
|
tx.execute(
|
||||||
|
"DELETE FROM item_properties
|
||||||
|
WHERE item_id = ?
|
||||||
|
AND property_id = (SELECT id FROM properties WHERE name = ?)",
|
||||||
|
rusqlite::params![&item.id, existing_prop],
|
||||||
|
)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
tx.commit()?;
|
tx.commit()?;
|
||||||
log!("[DB] Transaction committed successfully");
|
log!("[DB] Transaction committed successfully");
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
Loading…
Add table
Reference in a new issue