refactor: update api handlers to use async item service
Co-authored-by: aider (openai/andrew/openrouter/google/gemini-2.5-pro) <aider@aider.chat>
This commit is contained in:
@@ -6,6 +6,8 @@ use std::str::FromStr;
|
||||
use log::{debug, warn};
|
||||
|
||||
use crate::modes::server::common::AppState;
|
||||
use crate::core::async_item_service::AsyncItemService;
|
||||
use crate::core::error::CoreError;
|
||||
use crate::db;
|
||||
use crate::compression_engine::{CompressionType, get_compression_engine};
|
||||
use crate::meta_plugin::{MetaPluginType, get_meta_plugin};
|
||||
@@ -136,34 +138,27 @@ impl KeepTools {
|
||||
.and_then(|v| v.as_i64())
|
||||
.ok_or_else(|| ToolError::InvalidArguments("Missing or invalid 'id' field".to_string()))?;
|
||||
|
||||
let mut conn = self.state.db.lock().await;
|
||||
|
||||
let item = db::get_item(&mut *conn, item_id)?
|
||||
.ok_or_else(|| ToolError::InvalidArguments(format!("Item {} not found", item_id)))?;
|
||||
|
||||
// Get content
|
||||
let mut item_path = self.state.data_dir.clone();
|
||||
item_path.push(item_id.to_string());
|
||||
|
||||
let compression_type = CompressionType::from_str(&item.compression)?;
|
||||
let compression_engine = get_compression_engine(compression_type)?;
|
||||
|
||||
let mut reader = compression_engine.open(item_path)?;
|
||||
let mut content = String::new();
|
||||
reader.read_to_string(&mut content)?;
|
||||
|
||||
// Get metadata and tags
|
||||
let tags = db::get_item_tags(&mut *conn, &item)?;
|
||||
let metadata = db::get_item_meta(&mut *conn, &item)?;
|
||||
|
||||
let service = AsyncItemService::new(self.state.data_dir.clone(), self.state.db.clone());
|
||||
|
||||
let item_with_content = match service.get_item_content(item_id).await {
|
||||
Ok(iwc) => iwc,
|
||||
Err(CoreError::ItemNotFound(_)) => return Err(ToolError::InvalidArguments(format!("Item {} not found", item_id))),
|
||||
Err(e) => return Err(ToolError::Other(anyhow!(e))),
|
||||
};
|
||||
|
||||
let item = item_with_content.item_with_meta.item;
|
||||
let content = String::from_utf8_lossy(&item_with_content.content).to_string();
|
||||
let tags: Vec<String> = item_with_content.item_with_meta.tags.into_iter().map(|t| t.name).collect();
|
||||
let metadata = item_with_content.item_with_meta.meta_as_map();
|
||||
|
||||
let response = serde_json::json!({
|
||||
"id": item_id,
|
||||
"content": content,
|
||||
"timestamp": item.ts.to_rfc3339(),
|
||||
"size": item.size,
|
||||
"compression": item.compression,
|
||||
"tags": tags.iter().map(|t| &t.name).collect::<Vec<_>>(),
|
||||
"metadata": metadata.iter().map(|m| (&m.name, &m.value)).collect::<HashMap<_, _>>()
|
||||
"tags": tags,
|
||||
"metadata": metadata,
|
||||
});
|
||||
|
||||
Ok(serde_json::to_string_pretty(&response)?)
|
||||
@@ -176,40 +171,30 @@ impl KeepTools {
|
||||
.map(|arr| arr.iter().filter_map(|v| v.as_str().map(|s| s.to_string())).collect())
|
||||
.unwrap_or_default();
|
||||
|
||||
let mut conn = self.state.db.lock().await;
|
||||
|
||||
let item = if tags.is_empty() {
|
||||
db::get_item_last(&mut *conn)?
|
||||
} else {
|
||||
db::get_item_matching(&mut *conn, &tags, &HashMap::new())?
|
||||
let service = AsyncItemService::new(self.state.data_dir.clone(), self.state.db.clone());
|
||||
|
||||
let item_with_meta = match service.find_item(vec![], tags, HashMap::new()).await {
|
||||
Ok(iwm) => iwm,
|
||||
Err(CoreError::ItemNotFoundGeneric) => return Err(ToolError::InvalidArguments("No items found".to_string())),
|
||||
Err(e) => return Err(ToolError::Other(anyhow!(e))),
|
||||
};
|
||||
|
||||
let item = item.ok_or_else(|| ToolError::InvalidArguments("No items found".to_string()))?;
|
||||
let item_id = item.id.ok_or_else(|| anyhow!("Item missing ID"))?;
|
||||
|
||||
// Get content
|
||||
let mut item_path = self.state.data_dir.clone();
|
||||
item_path.push(item_id.to_string());
|
||||
|
||||
let compression_type = CompressionType::from_str(&item.compression)?;
|
||||
let compression_engine = get_compression_engine(compression_type)?;
|
||||
|
||||
let mut reader = compression_engine.open(item_path)?;
|
||||
let mut content = String::new();
|
||||
reader.read_to_string(&mut content)?;
|
||||
|
||||
// Get metadata and tags
|
||||
let tags = db::get_item_tags(&mut *conn, &item)?;
|
||||
let metadata = db::get_item_meta(&mut *conn, &item)?;
|
||||
let item_id = item_with_meta.item.id.ok_or_else(|| anyhow!("Item missing ID after find"))?;
|
||||
let item_with_content = service.get_item_content(item_id).await.map_err(|e| ToolError::Other(anyhow!(e)))?;
|
||||
|
||||
let item = item_with_content.item_with_meta.item;
|
||||
let content = String::from_utf8_lossy(&item_with_content.content).to_string();
|
||||
let tags: Vec<String> = item_with_content.item_with_meta.tags.into_iter().map(|t| t.name).collect();
|
||||
let metadata = item_with_content.item_with_meta.meta_as_map();
|
||||
|
||||
let response = serde_json::json!({
|
||||
"id": item_id,
|
||||
"content": content,
|
||||
"timestamp": item.ts.to_rfc3339(),
|
||||
"size": item.size,
|
||||
"compression": item.compression,
|
||||
"tags": tags.iter().map(|t| &t.name).collect::<Vec<_>>(),
|
||||
"metadata": metadata.iter().map(|m| (&m.name, &m.value)).collect::<HashMap<_, _>>()
|
||||
"tags": tags,
|
||||
"metadata": metadata,
|
||||
});
|
||||
|
||||
Ok(serde_json::to_string_pretty(&response)?)
|
||||
@@ -233,36 +218,20 @@ impl KeepTools {
|
||||
.and_then(|v| v.as_u64())
|
||||
.unwrap_or(0) as usize;
|
||||
|
||||
let mut conn = self.state.db.lock().await;
|
||||
|
||||
let items = if tags.is_empty() {
|
||||
db::get_items(&mut *conn)?
|
||||
} else {
|
||||
db::get_items_matching(&mut *conn, &tags, &HashMap::new())?
|
||||
};
|
||||
|
||||
let service = AsyncItemService::new(self.state.data_dir.clone(), self.state.db.clone());
|
||||
let mut items_with_meta = service.list_items(tags, HashMap::new()).await.map_err(|e| ToolError::Other(anyhow!(e)))?;
|
||||
|
||||
// Sort by timestamp (newest first) and apply pagination
|
||||
let mut items = items;
|
||||
items.sort_by(|a, b| b.ts.cmp(&a.ts));
|
||||
let items: Vec<_> = items.into_iter().skip(offset).take(limit).collect();
|
||||
items_with_meta.sort_by(|a, b| b.item.ts.cmp(&a.item.ts));
|
||||
let items_with_meta: Vec<_> = items_with_meta.into_iter().skip(offset).take(limit).collect();
|
||||
|
||||
// Get item IDs for batch queries
|
||||
let item_ids: Vec<i64> = items.iter().filter_map(|item| item.id).collect();
|
||||
|
||||
// Get tags and metadata for all items
|
||||
let tags_map = db::get_tags_for_items(&mut *conn, &item_ids)?;
|
||||
let meta_map = db::get_meta_for_items(&mut *conn, &item_ids)?;
|
||||
|
||||
let items_info: Vec<_> = items
|
||||
let items_info: Vec<_> = items_with_meta
|
||||
.into_iter()
|
||||
.map(|item| {
|
||||
.map(|item_with_meta| {
|
||||
let item = item_with_meta.item;
|
||||
let item_id = item.id.unwrap_or(0);
|
||||
let item_tags = tags_map.get(&item_id)
|
||||
.map(|tags| tags.iter().map(|t| &t.name).collect::<Vec<_>>())
|
||||
.unwrap_or_default();
|
||||
let item_meta = meta_map.get(&item_id)
|
||||
.cloned()
|
||||
.unwrap_or_default();
|
||||
let item_tags: Vec<String> = item_with_meta.tags.into_iter().map(|t| t.name).collect();
|
||||
let item_meta = item_with_meta.meta_as_map();
|
||||
|
||||
serde_json::json!({
|
||||
"id": item_id,
|
||||
@@ -302,31 +271,19 @@ impl KeepTools {
|
||||
}).collect())
|
||||
.unwrap_or_default();
|
||||
|
||||
let mut conn = self.state.db.lock().await;
|
||||
|
||||
let items = db::get_items_matching(&mut *conn, &tags, &metadata)?;
|
||||
|
||||
let service = AsyncItemService::new(self.state.data_dir.clone(), self.state.db.clone());
|
||||
let mut items_with_meta = service.list_items(tags.clone(), metadata.clone()).await.map_err(|e| ToolError::Other(anyhow!(e)))?;
|
||||
|
||||
// Sort by timestamp (newest first)
|
||||
let mut items = items;
|
||||
items.sort_by(|a, b| b.ts.cmp(&a.ts));
|
||||
items_with_meta.sort_by(|a, b| b.item.ts.cmp(&a.item.ts));
|
||||
|
||||
// Get item IDs for batch queries
|
||||
let item_ids: Vec<i64> = items.iter().filter_map(|item| item.id).collect();
|
||||
|
||||
// Get tags and metadata for all items
|
||||
let tags_map = db::get_tags_for_items(&mut *conn, &item_ids)?;
|
||||
let meta_map = db::get_meta_for_items(&mut *conn, &item_ids)?;
|
||||
|
||||
let items_info: Vec<_> = items
|
||||
let items_info: Vec<_> = items_with_meta
|
||||
.into_iter()
|
||||
.map(|item| {
|
||||
.map(|item_with_meta| {
|
||||
let item = item_with_meta.item;
|
||||
let item_id = item.id.unwrap_or(0);
|
||||
let item_tags = tags_map.get(&item_id)
|
||||
.map(|tags| tags.iter().map(|t| &t.name).collect::<Vec<_>>())
|
||||
.unwrap_or_default();
|
||||
let item_meta = meta_map.get(&item_id)
|
||||
.cloned()
|
||||
.unwrap_or_default();
|
||||
let item_tags: Vec<String> = item_with_meta.tags.into_iter().map(|t| t.name).collect();
|
||||
let item_meta = item_with_meta.meta_as_map();
|
||||
|
||||
serde_json::json!({
|
||||
"id": item_id,
|
||||
|
||||
Reference in New Issue
Block a user