use crate::entity_type::EntityType;
use crate::errors::AppError;
use crate::i18n::errors_msg;
use crate::output::{self, OutputFormat};
use crate::paths::AppPaths;
use crate::storage::connection::open_rw;
use crate::storage::entities;
use rusqlite::params;
use serde::Serialize;
#[derive(clap::Args)]
#[command(after_long_help = "EXAMPLES:\n \
# Rename an entity\n \
sqlite-graphrag rename-entity --name old-name --new-name new-name\n\n \
# Rename with namespace\n \
sqlite-graphrag rename-entity --name auth --new-name authentication --namespace my-project")]
pub struct RenameEntityArgs {
#[arg(long, value_name = "NAME")]
pub name: String,
#[arg(long, value_name = "NEW_NAME")]
pub new_name: String,
#[arg(long)]
pub namespace: Option<String>,
#[arg(long, value_enum, default_value = "json")]
pub format: OutputFormat,
#[arg(long, hide = true, help = "No-op; JSON is always emitted on stdout")]
pub json: bool,
#[arg(long, env = "SQLITE_GRAPHRAG_DB_PATH")]
pub db: Option<String>,
}
#[derive(Serialize)]
struct RenameEntityResponse {
action: String,
old_name: String,
new_name: String,
entity_id: i64,
namespace: String,
elapsed_ms: u64,
}
pub fn run(args: RenameEntityArgs) -> Result<(), AppError> {
let start = std::time::Instant::now();
let namespace = crate::namespace::resolve_namespace(args.namespace.as_deref())?;
let paths = AppPaths::resolve(args.db.as_deref())?;
crate::storage::connection::ensure_db_ready(&paths)?;
let mut conn = open_rw(&paths.db)?;
let row: Option<(i64, EntityType)> = {
let mut stmt = conn
.prepare_cached("SELECT id, type FROM entities WHERE namespace = ?1 AND name = ?2")?;
match stmt.query_row(params![namespace, args.name], |r| {
Ok((r.get::<_, i64>(0)?, r.get::<_, EntityType>(1)?))
}) {
Ok(row) => Some(row),
Err(rusqlite::Error::QueryReturnedNoRows) => None,
Err(e) => return Err(AppError::Database(e)),
}
};
let (entity_id, entity_type) = row
.ok_or_else(|| AppError::NotFound(errors_msg::entity_not_found(&args.name, &namespace)))?;
if entities::find_entity_id(&conn, &namespace, &args.new_name)?.is_some() {
return Err(AppError::Validation(format!(
"entity with name '{}' already exists in namespace '{}'",
args.new_name, namespace
)));
}
let embedding = crate::daemon::embed_passage_or_local(&paths.models, &args.new_name)?;
let tx = conn.transaction_with_behavior(rusqlite::TransactionBehavior::Immediate)?;
tx.execute(
"UPDATE entities SET name = ?1, updated_at = unixepoch() WHERE id = ?2",
params![args.new_name, entity_id],
)?;
tx.execute(
"DELETE FROM vec_entities WHERE entity_id = ?1",
params![entity_id],
)?;
let embedding_bytes = crate::embedder::f32_to_bytes(&embedding);
tx.execute(
"INSERT INTO vec_entities(entity_id, namespace, type, embedding, name)
VALUES (?1, ?2, ?3, ?4, ?5)",
params![
entity_id,
namespace,
entity_type,
&embedding_bytes,
args.new_name
],
)?;
tx.commit()?;
conn.execute_batch("PRAGMA wal_checkpoint(TRUNCATE);")?;
let response = RenameEntityResponse {
action: "renamed".to_string(),
old_name: args.name,
new_name: args.new_name,
entity_id,
namespace: namespace.clone(),
elapsed_ms: start.elapsed().as_millis() as u64,
};
match args.format {
OutputFormat::Json => output::emit_json(&response)?,
OutputFormat::Text | OutputFormat::Markdown => {
output::emit_text(&format!(
"renamed entity: '{}' → '{}' [{}]",
response.old_name, response.new_name, response.namespace
));
}
}
Ok(())
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn rename_entity_response_serializes_all_fields() {
let resp = RenameEntityResponse {
action: "renamed".to_string(),
old_name: "auth".to_string(),
new_name: "authentication".to_string(),
entity_id: 42,
namespace: "global".to_string(),
elapsed_ms: 7,
};
let json = serde_json::to_value(&resp).expect("serialization failed");
assert_eq!(json["action"], "renamed");
assert_eq!(json["old_name"], "auth");
assert_eq!(json["new_name"], "authentication");
assert_eq!(json["entity_id"], 42);
assert_eq!(json["namespace"], "global");
assert!(json["elapsed_ms"].is_number());
}
#[test]
fn rename_entity_response_action_is_renamed() {
let resp = RenameEntityResponse {
action: "renamed".to_string(),
old_name: "x".to_string(),
new_name: "y".to_string(),
entity_id: 1,
namespace: "ns".to_string(),
elapsed_ms: 1,
};
assert_eq!(resp.action, "renamed");
}
#[test]
fn rename_entity_response_entity_id_preserved() {
let resp = RenameEntityResponse {
action: "renamed".to_string(),
old_name: "old".to_string(),
new_name: "new".to_string(),
entity_id: 999,
namespace: "test-ns".to_string(),
elapsed_ms: 5,
};
let json = serde_json::to_value(&resp).expect("serialization failed");
assert_eq!(json["entity_id"], 999);
}
#[test]
fn rename_entity_response_namespace_reflected() {
let resp = RenameEntityResponse {
action: "renamed".to_string(),
old_name: "a".to_string(),
new_name: "b".to_string(),
entity_id: 10,
namespace: "my-project".to_string(),
elapsed_ms: 2,
};
let json = serde_json::to_value(&resp).expect("serialization failed");
assert_eq!(json["namespace"], "my-project");
}
}