2773 lines
85 KiB
Rust
2773 lines
85 KiB
Rust
use axum::{
|
|
body::Body,
|
|
extract::{Path, Query, State},
|
|
http::{header, StatusCode},
|
|
response::IntoResponse,
|
|
response::Json,
|
|
routing::{get, patch, post},
|
|
Router,
|
|
};
|
|
use serde::{Deserialize, Serialize};
|
|
use uuid::Uuid;
|
|
|
|
use crate::core::db::{Database, PostgresDb};
|
|
use crate::core::person_identity::{
|
|
ChunkPersonInfo, CreatePersonIdentityRequest, PersonIdentity, PersonIdentityResponse,
|
|
PersonMatch, PersonStatistics, PersonTimelineEntry, PersonTimelineResponse,
|
|
UpdatePersonIdentityRequest,
|
|
};
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct IdentifyPersonsRequest {
|
|
pub video_uuid: String,
|
|
pub auto_match: Option<bool>,
|
|
pub match_threshold: Option<f64>,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct IdentifyPersonsResponse {
|
|
pub success: bool,
|
|
pub message: String,
|
|
pub persons: Vec<PersonIdentityResponse>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct PersonTimelineQuery {
|
|
pub video_uuid: String,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct FaceThumbnailQuery {
|
|
pub video_uuid: String,
|
|
#[serde(default)]
|
|
pub index: Option<usize>, // Which face detection to use (default: 0)
|
|
}
|
|
|
|
// Structs for parsing face_clustered.json
|
|
#[derive(Debug, Deserialize)]
|
|
struct FaceDetection {
|
|
#[serde(default)]
|
|
person_id: Option<String>,
|
|
x: i32,
|
|
y: i32,
|
|
width: i32,
|
|
height: i32,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
struct FaceFrame {
|
|
timestamp: f64,
|
|
faces: Vec<FaceDetection>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
struct FaceClusteredData {
|
|
frames: Vec<FaceFrame>,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct ChunkPersonsResponse {
|
|
pub success: bool,
|
|
pub chunk_id: String,
|
|
pub persons: Vec<ChunkPersonInfo>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct MergePersonsRequest {
|
|
pub video_uuid: String,
|
|
pub target_person_id: String,
|
|
pub source_person_ids: Vec<String>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct UndoMergeRequest {
|
|
pub merge_id: String,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct MergeHistoryEntry {
|
|
pub merge_id: String,
|
|
pub target_person_id: String,
|
|
pub source_person_ids: Vec<String>,
|
|
pub original_target_stats: serde_json::Value,
|
|
pub original_source_stats: serde_json::Value,
|
|
pub merged_at: String,
|
|
pub is_undone: bool,
|
|
pub undone_at: Option<String>,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct MergeHistoryResponse {
|
|
pub success: bool,
|
|
pub history: Vec<MergeHistoryEntry>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct PersonListQuery {
|
|
pub video_uuid: String,
|
|
pub limit: Option<i32>,
|
|
pub offset: Option<i32>,
|
|
pub min_appearances: Option<i32>,
|
|
pub has_speaker: Option<bool>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct AutoIdentifyRequest {
|
|
pub video_uuid: String,
|
|
pub min_speaker_confidence: Option<f64>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct SimilarPersonsQuery {
|
|
pub video_uuid: String,
|
|
pub threshold: Option<f64>,
|
|
pub limit: Option<i32>,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct NamingSuggestion {
|
|
pub person_id: String,
|
|
pub current_name: Option<String>,
|
|
pub suggested_name: String,
|
|
pub confidence: f64,
|
|
pub sources: Vec<SuggestionSource>,
|
|
pub action: String, // "auto_apply" or "needs_review"
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct SuggestionSource {
|
|
pub r#type: String, // "speaker_match", "talent_db", "ocr_context", "face_similarity"
|
|
pub detail: String,
|
|
pub weight: f64,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct MergeSuggestion {
|
|
pub person_id: String,
|
|
pub merge_with: Vec<String>,
|
|
pub confidence: f64,
|
|
pub reasons: Vec<String>,
|
|
pub action: String, // "auto_apply" or "needs_review"
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct SuggestionsResponse {
|
|
pub success: bool,
|
|
pub naming_suggestions: Vec<NamingSuggestion>,
|
|
pub merge_suggestions: Vec<MergeSuggestion>,
|
|
pub total_naming: usize,
|
|
pub total_merge: usize,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct PersonSummary {
|
|
pub person_id: String,
|
|
pub name: Option<String>,
|
|
pub speaker_id: Option<String>,
|
|
pub appearance_count: i32,
|
|
pub total_appearance_duration: f64,
|
|
pub first_appearance_time: Option<f64>,
|
|
pub last_appearance_time: Option<f64>,
|
|
pub is_confirmed: bool,
|
|
pub speaker_confidence: Option<f64>,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct PersonListResponse {
|
|
pub success: bool,
|
|
pub persons: Vec<PersonSummary>,
|
|
pub total: i64,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct MergePersonsResponse {
|
|
pub success: bool,
|
|
pub message: String,
|
|
pub target_person_id: String,
|
|
pub merge_id: String,
|
|
}
|
|
|
|
#[derive(Debug, Serialize)]
|
|
pub struct AutoIdentifyResponse {
|
|
pub success: bool,
|
|
pub message: String,
|
|
pub total_persons: i32,
|
|
pub matched_speakers: i32,
|
|
pub persons: Vec<PersonSummary>,
|
|
}
|
|
|
|
pub fn person_identity_routes() -> Router<crate::api::server::AppState> {
|
|
Router::new()
|
|
.route("/api/v1/person/identify", post(identify_persons))
|
|
.route("/api/v1/person/auto-identify", post(auto_identify_persons))
|
|
.route("/api/v1/person/suggest", post(get_person_suggestions))
|
|
.route("/api/v1/person/list", get(list_persons))
|
|
.route("/api/v1/person/merge", post(merge_persons))
|
|
.route("/api/v1/person/merge/undo", post(undo_merge))
|
|
.route("/api/v1/person/merge/history", get(get_merge_history))
|
|
.route(
|
|
"/api/v1/person/:person_id/unbind-speaker",
|
|
post(unbind_speaker),
|
|
)
|
|
.route(
|
|
"/api/v1/person/:person_id/reassign-speaker",
|
|
post(reassign_speaker),
|
|
)
|
|
.route(
|
|
"/api/v1/person/:person_id/remove-appearance",
|
|
post(remove_appearance),
|
|
)
|
|
.route(
|
|
"/api/v1/person/:person_id/reassign-appearance",
|
|
post(reassign_appearance),
|
|
)
|
|
.route("/api/v1/person/:person_id/split", post(split_person))
|
|
.route(
|
|
"/api/v1/person/:person_id/similar",
|
|
get(get_similar_persons),
|
|
)
|
|
.route(
|
|
"/api/v1/person/:person_id/confirm",
|
|
patch(confirm_person_suggestion),
|
|
)
|
|
.route("/api/v1/person/:person_id", get(get_person_details))
|
|
.route("/api/v1/person/:person_id", patch(update_person_identity))
|
|
.route(
|
|
"/api/v1/person/:person_id/timeline",
|
|
get(get_person_timeline),
|
|
)
|
|
.route(
|
|
"/api/v1/person/:person_id/appearances",
|
|
get(get_person_appearances),
|
|
)
|
|
.route(
|
|
"/api/v1/person/:person_id/thumbnail",
|
|
get(get_person_thumbnail),
|
|
)
|
|
.route("/api/v1/chunks/:chunk_id/persons", get(get_chunk_persons))
|
|
}
|
|
|
|
async fn identify_persons(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Json(request): Json<IdentifyPersonsRequest>,
|
|
) -> Result<Json<IdentifyPersonsResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to connect to database: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
tracing::info!(
|
|
"[PERSON_IDENTITY] Identifying persons for video: {}",
|
|
request.video_uuid
|
|
);
|
|
|
|
let auto_match = request.auto_match.unwrap_or(true);
|
|
let threshold = request.match_threshold.unwrap_or(0.5);
|
|
|
|
if auto_match {
|
|
let matches = match auto_match_face_speaker(&db, &request.video_uuid, threshold).await {
|
|
Ok(m) => m,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to auto-match: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let mut persons = Vec::new();
|
|
for match_result in matches {
|
|
let person = match create_person_identity(
|
|
&db,
|
|
CreatePersonIdentityRequest {
|
|
video_uuid: request.video_uuid.clone(),
|
|
face_identity_id: None,
|
|
speaker_id: Some(match_result.speaker_id.clone()),
|
|
name: None,
|
|
metadata: Some(serde_json::json!({
|
|
"auto_matched": true,
|
|
"confidence": match_result.confidence,
|
|
"match_count": match_result.match_count
|
|
})),
|
|
},
|
|
)
|
|
.await
|
|
{
|
|
Ok(p) => p,
|
|
Err(e) => {
|
|
tracing::warn!("Failed to create person identity: {}", e);
|
|
continue;
|
|
}
|
|
};
|
|
|
|
persons.push(PersonIdentityResponse::from(person));
|
|
}
|
|
|
|
Ok(Json(IdentifyPersonsResponse {
|
|
success: true,
|
|
message: format!("Identified {} persons", persons.len()),
|
|
persons,
|
|
}))
|
|
} else {
|
|
Ok(Json(IdentifyPersonsResponse {
|
|
success: true,
|
|
message: "Auto-match disabled, no persons identified".to_string(),
|
|
persons: vec![],
|
|
}))
|
|
}
|
|
}
|
|
|
|
async fn get_person_details(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to connect to database: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let query = r#"
|
|
SELECT
|
|
person_id, name, face_identity_id, speaker_id,
|
|
confidence, appearance_count, total_appearance_duration,
|
|
first_appearance_time, last_appearance_time,
|
|
is_confirmed, created_at, updated_at
|
|
FROM person_identities
|
|
WHERE person_id = $1
|
|
"#;
|
|
|
|
let person: Option<(
|
|
String,
|
|
Option<String>,
|
|
Option<i32>,
|
|
Option<String>,
|
|
f64,
|
|
i32,
|
|
f64,
|
|
Option<f64>,
|
|
Option<f64>,
|
|
bool,
|
|
chrono::DateTime<chrono::Utc>,
|
|
chrono::DateTime<chrono::Utc>,
|
|
)> = match sqlx::query_as(query)
|
|
.bind(&person_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(person) => person,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch person: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match person {
|
|
Some(person_data) => {
|
|
let response = serde_json::json!({
|
|
"success": true,
|
|
"person_id": person_data.0,
|
|
"name": person_data.1,
|
|
"face_identity_id": person_data.2,
|
|
"speaker_id": person_data.3,
|
|
"confidence": person_data.4,
|
|
"appearance_count": person_data.5,
|
|
"total_appearance_duration": person_data.6,
|
|
"first_appearance_time": person_data.7,
|
|
"last_appearance_time": person_data.8,
|
|
"is_confirmed": person_data.9,
|
|
"created_at": person_data.10.to_rfc3339(),
|
|
"updated_at": person_data.11.to_rfc3339()
|
|
});
|
|
|
|
Ok(Json(response))
|
|
}
|
|
None => Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
)),
|
|
}
|
|
}
|
|
|
|
async fn update_person_identity(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<UpdatePersonIdentityRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to connect to database: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
tracing::info!("[PERSON_IDENTITY] Updating person: {}", person_id);
|
|
|
|
let query = r#"
|
|
UPDATE person_identities
|
|
SET
|
|
name = COALESCE($2, name),
|
|
metadata = COALESCE($3, metadata),
|
|
is_confirmed = COALESCE($4, is_confirmed),
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $1
|
|
RETURNING person_id, name
|
|
"#;
|
|
|
|
let updated: Option<(String, Option<String>)> = match sqlx::query_as(query)
|
|
.bind(&person_id)
|
|
.bind(&request.name)
|
|
.bind(&request.metadata)
|
|
.bind(&request.is_confirmed)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(result) => result,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to update person: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match updated {
|
|
Some((id, name)) => {
|
|
let response = serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Person '{}' updated successfully", name.unwrap_or_else(|| id.clone())),
|
|
"person_id": id
|
|
});
|
|
|
|
Ok(Json(response))
|
|
}
|
|
None => Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
)),
|
|
}
|
|
}
|
|
|
|
async fn get_person_timeline(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Query(query): Query<PersonTimelineQuery>,
|
|
) -> Result<Json<PersonTimelineResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to connect to database: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let name_query = "SELECT name FROM person_identities WHERE person_id = $1";
|
|
let name: Option<String> = match sqlx::query_scalar(name_query)
|
|
.bind(&person_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(name) => name,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch person name: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let timeline_query = r#"
|
|
SELECT start_time, end_time, duration, confidence
|
|
FROM person_appearances
|
|
WHERE person_id = $1 AND video_uuid = $2
|
|
ORDER BY start_time ASC
|
|
"#;
|
|
|
|
let timeline: Vec<(f64, f64, f64, f64)> = match sqlx::query_as(timeline_query)
|
|
.bind(&person_id)
|
|
.bind(&query.video_uuid)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
{
|
|
Ok(timeline) => timeline,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch timeline: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let timeline: Vec<PersonTimelineEntry> = timeline
|
|
.into_iter()
|
|
.map(|(start, end, duration, confidence)| PersonTimelineEntry {
|
|
start_time: start,
|
|
end_time: end,
|
|
duration,
|
|
confidence,
|
|
})
|
|
.collect();
|
|
|
|
let stats_query = r#"
|
|
SELECT
|
|
COUNT(*) as total_appearances,
|
|
SUM(duration) as total_duration,
|
|
MIN(start_time) as first_appearance,
|
|
MAX(end_time) as last_appearance,
|
|
AVG(confidence) as average_confidence
|
|
FROM person_appearances
|
|
WHERE person_id = $1 AND video_uuid = $2
|
|
"#;
|
|
|
|
let stats: (i64, Option<f64>, Option<f64>, Option<f64>, Option<f64>) =
|
|
match sqlx::query_as(stats_query)
|
|
.bind(&person_id)
|
|
.bind(&query.video_uuid)
|
|
.fetch_one(db.pool())
|
|
.await
|
|
{
|
|
Ok(stats) => stats,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch statistics: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let statistics = PersonStatistics {
|
|
total_appearances: stats.0 as i32,
|
|
total_duration: stats.1.unwrap_or(0.0),
|
|
first_appearance: stats.2,
|
|
last_appearance: stats.3,
|
|
average_confidence: stats.4.unwrap_or(0.0),
|
|
};
|
|
|
|
Ok(Json(PersonTimelineResponse {
|
|
person_id,
|
|
name,
|
|
timeline,
|
|
statistics,
|
|
}))
|
|
}
|
|
|
|
async fn get_person_appearances(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to connect to database: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let query = r#"
|
|
SELECT
|
|
person_id, video_uuid, start_time, end_time, duration,
|
|
face_detection_id, asrx_segment_start, asrx_segment_end,
|
|
confidence, created_at
|
|
FROM person_appearances
|
|
WHERE person_id = $1
|
|
ORDER BY start_time DESC
|
|
LIMIT 100
|
|
"#;
|
|
|
|
let appearances: Vec<serde_json::Value> = match sqlx::query(query)
|
|
.bind(&person_id)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
{
|
|
Ok(rows) => {
|
|
rows.iter()
|
|
.map(|row| {
|
|
use sqlx::Row;
|
|
serde_json::json!({
|
|
"person_id": row.get::<String, _>("person_id"),
|
|
"video_uuid": row.get::<String, _>("video_uuid"),
|
|
"start_time": row.get::<f64, _>("start_time"),
|
|
"end_time": row.get::<f64, _>("end_time"),
|
|
"duration": row.get::<f64, _>("duration"),
|
|
"confidence": row.get::<f64, _>("confidence"),
|
|
"created_at": row.get::<chrono::DateTime<chrono::Utc>, _>("created_at").to_rfc3339()
|
|
})
|
|
})
|
|
.collect()
|
|
}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch appearances: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"person_id": person_id,
|
|
"appearances": appearances
|
|
})))
|
|
}
|
|
|
|
async fn get_chunk_persons(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(chunk_id): Path<String>,
|
|
) -> Result<Json<ChunkPersonsResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to connect to database: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let chunk_query = r#"
|
|
SELECT uuid, start_time, end_time, metadata
|
|
FROM chunks
|
|
WHERE chunk_id = $1
|
|
"#;
|
|
|
|
let chunk: Option<(String, f64, f64, Option<serde_json::Value>)> =
|
|
match sqlx::query_as(chunk_query)
|
|
.bind(&chunk_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(chunk) => chunk,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch chunk: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let chunk = match chunk {
|
|
Some(c) => c,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Chunk not found: {}", chunk_id),
|
|
))
|
|
}
|
|
};
|
|
|
|
let (video_uuid, start_time, end_time, _metadata) = chunk;
|
|
|
|
let persons_query = r#"
|
|
SELECT
|
|
pi.person_id,
|
|
pi.name,
|
|
pa.confidence,
|
|
LEAST(pa.end_time, $3) - GREATEST(pa.start_time, $2) as overlap_duration
|
|
FROM person_appearances pa
|
|
JOIN person_identities pi ON pa.person_id = pi.person_id
|
|
WHERE pa.video_uuid = $1
|
|
AND pa.start_time < $3
|
|
AND pa.end_time > $2
|
|
ORDER BY overlap_duration DESC
|
|
"#;
|
|
|
|
let persons: Vec<ChunkPersonInfo> =
|
|
match sqlx::query_as::<_, (String, Option<String>, f64, f64)>(persons_query)
|
|
.bind(&video_uuid)
|
|
.bind(start_time)
|
|
.bind(end_time)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
{
|
|
Ok(rows) => rows
|
|
.into_iter()
|
|
.map(
|
|
|(person_id, name, confidence, overlap_duration)| ChunkPersonInfo {
|
|
person_id,
|
|
name,
|
|
confidence,
|
|
overlap_duration,
|
|
},
|
|
)
|
|
.collect(),
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Failed to fetch persons: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(ChunkPersonsResponse {
|
|
success: true,
|
|
chunk_id,
|
|
persons,
|
|
}))
|
|
}
|
|
|
|
/// Extracts a face thumbnail for a given person from the video
|
|
async fn get_person_thumbnail(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Query(query): Query<FaceThumbnailQuery>,
|
|
) -> Result<impl IntoResponse, (StatusCode, String)> {
|
|
// 1. Locate the face_clustered.json file
|
|
let json_path = format!(
|
|
"output/{}/{}_face_clustered.json",
|
|
query.video_uuid, query.video_uuid
|
|
);
|
|
let json_path2 = format!(
|
|
"output/{}/{}.face_clustered.json",
|
|
query.video_uuid, query.video_uuid
|
|
);
|
|
|
|
// Fallback path if the naming convention is slightly different
|
|
let fallback_path = format!("output/{}/face_clustered.json", query.video_uuid);
|
|
|
|
let path = if std::path::Path::new(&json_path).exists() {
|
|
json_path
|
|
} else if std::path::Path::new(&json_path2).exists() {
|
|
json_path2
|
|
} else if std::path::Path::new(&fallback_path).exists() {
|
|
fallback_path
|
|
} else {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!(
|
|
"Face data not found for video: {}. Tried: {}, {}, {}",
|
|
query.video_uuid, json_path, json_path2, fallback_path
|
|
),
|
|
));
|
|
};
|
|
|
|
// 2. Parse the JSON to find the person's face
|
|
let content = match tokio::fs::read_to_string(&path).await {
|
|
Ok(c) => c,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Read error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let data: FaceClusteredData = match serde_json::from_str(&content) {
|
|
Ok(d) => d,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Parse error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let mut detections = Vec::new();
|
|
for frame in &data.frames {
|
|
for face in &frame.faces {
|
|
if let Some(pid) = &face.person_id {
|
|
if pid == &person_id {
|
|
detections.push((frame.timestamp, face));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if detections.is_empty() {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("No detections found for person: {}", person_id),
|
|
));
|
|
}
|
|
|
|
let index = query.index.unwrap_or(0).min(detections.len() - 1);
|
|
let (timestamp, face) = detections[index];
|
|
|
|
// 3. Locate the video file
|
|
let video_path = format!("output/{}/{}.mp4", query.video_uuid, query.video_uuid);
|
|
if !std::path::Path::new(&video_path).exists() {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Video file not found: {}", video_path),
|
|
));
|
|
}
|
|
|
|
// 4. Use ffmpeg to extract and crop the face
|
|
// ffmpeg -ss {timestamp} -i {video} -vf "crop=w:h:x:y" -frames:v 1 -f image2pipe -vcodec mjpeg -
|
|
let crop_filter = format!("crop={}:{}:{}:{}", face.width, face.height, face.x, face.y);
|
|
|
|
let output = match tokio::process::Command::new("ffmpeg")
|
|
.args(&[
|
|
"-ss",
|
|
×tamp.to_string(),
|
|
"-i",
|
|
&video_path,
|
|
"-vf",
|
|
&crop_filter,
|
|
"-frames:v",
|
|
"1",
|
|
"-f",
|
|
"image2pipe",
|
|
"-vcodec",
|
|
"mjpeg",
|
|
"-",
|
|
])
|
|
.output()
|
|
.await
|
|
{
|
|
Ok(o) => o,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("ffmpeg error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
if !output.status.success() {
|
|
let stderr = String::from_utf8_lossy(&output.stderr);
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("ffmpeg failed: {}", stderr),
|
|
));
|
|
}
|
|
|
|
// 5. Return the image
|
|
let response = axum::response::Response::builder()
|
|
.status(StatusCode::OK)
|
|
.header(header::CONTENT_TYPE, "image/jpeg")
|
|
.body(Body::from(output.stdout))
|
|
.map_err(|e| {
|
|
(
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Response error: {}", e),
|
|
)
|
|
})?;
|
|
|
|
Ok(response)
|
|
}
|
|
|
|
async fn create_person_identity(
|
|
db: &PostgresDb,
|
|
request: CreatePersonIdentityRequest,
|
|
) -> Result<PersonIdentity, anyhow::Error> {
|
|
let person_id = format!("person_{}", Uuid::new_v4());
|
|
|
|
let query = r#"
|
|
INSERT INTO person_identities (
|
|
person_id, video_uuid, face_identity_id, speaker_id,
|
|
name, metadata, confidence
|
|
) VALUES ($1, $2, $3, $4, $5, $6, $7)
|
|
RETURNING
|
|
id, person_id, face_identity_id, speaker_id,
|
|
video_uuid, confidence, name, metadata,
|
|
first_appearance_time, last_appearance_time,
|
|
total_appearance_duration, appearance_count,
|
|
created_at, updated_at, is_confirmed
|
|
"#;
|
|
|
|
let person: PersonIdentity = sqlx::query_as(query)
|
|
.bind(&person_id)
|
|
.bind(&request.video_uuid)
|
|
.bind(&request.face_identity_id)
|
|
.bind(&request.speaker_id)
|
|
.bind(&request.name)
|
|
.bind(&request.metadata.unwrap_or(serde_json::json!({})))
|
|
.bind(0.0)
|
|
.fetch_one(db.pool())
|
|
.await?;
|
|
|
|
Ok(person)
|
|
}
|
|
|
|
async fn auto_match_face_speaker(
|
|
db: &PostgresDb,
|
|
video_uuid: &str,
|
|
threshold: f64,
|
|
) -> Result<Vec<PersonMatch>, anyhow::Error> {
|
|
let query = "SELECT * FROM auto_match_face_speaker($1, $2)";
|
|
|
|
let matches: Vec<PersonMatch> = sqlx::query_as(query)
|
|
.bind(video_uuid)
|
|
.bind(threshold)
|
|
.fetch_all(db.pool())
|
|
.await?;
|
|
|
|
Ok(matches)
|
|
}
|
|
|
|
/// Auto-identify persons from face_clustered.json + ASRX speaker data
|
|
async fn auto_identify_persons(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Json(request): Json<AutoIdentifyRequest>,
|
|
) -> Result<Json<AutoIdentifyResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// 1. Load face_clustered.json
|
|
let clustered_path = format!(
|
|
"output/{}/{}.face_clustered.json",
|
|
request.video_uuid, request.video_uuid
|
|
);
|
|
let fallback_path = format!("output/{}/face_clustered.json", request.video_uuid);
|
|
let path = if std::path::Path::new(&clustered_path).exists() {
|
|
clustered_path
|
|
} else if std::path::Path::new(&fallback_path).exists() {
|
|
fallback_path
|
|
} else {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!(
|
|
"face_clustered.json not found for video: {}",
|
|
request.video_uuid
|
|
),
|
|
));
|
|
};
|
|
|
|
let content = match tokio::fs::read_to_string(&path).await {
|
|
Ok(c) => c,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Read error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let clustered: FaceClusteredData = match serde_json::from_str(&content) {
|
|
Ok(d) => d,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Parse error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// 2. Build person stats from face_clustered.json
|
|
use std::collections::HashMap;
|
|
#[derive(Default)]
|
|
struct PersonStat {
|
|
frame_count: i32,
|
|
first_time: Option<f64>,
|
|
last_time: Option<f64>,
|
|
timestamps: Vec<f64>,
|
|
}
|
|
|
|
let mut person_stats: HashMap<String, PersonStat> = HashMap::new();
|
|
for frame in &clustered.frames {
|
|
for face in &frame.faces {
|
|
if let Some(ref pid) = face.person_id {
|
|
let stat = person_stats.entry(pid.clone()).or_default();
|
|
stat.frame_count += 1;
|
|
stat.timestamps.push(frame.timestamp);
|
|
if stat.first_time.is_none() || Some(frame.timestamp) < stat.first_time {
|
|
stat.first_time = Some(frame.timestamp);
|
|
}
|
|
if stat.last_time.is_none() || Some(frame.timestamp) > stat.last_time {
|
|
stat.last_time = Some(frame.timestamp);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// 3. Load ASRX from chunks
|
|
let asrx_query = "SELECT chunk_id, content::text FROM chunks WHERE uuid = $1 AND chunk_type = 'trace' AND chunk_id LIKE 'trace_asrx_%'";
|
|
let asrx_chunks: Vec<(String, String)> = match sqlx::query_as(asrx_query)
|
|
.bind(&request.video_uuid)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
{
|
|
Ok(rows) => rows,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("ASRX query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Also check sentence chunks for speaker_id
|
|
let sentence_query = "SELECT content::text FROM chunks WHERE uuid = $1 AND chunk_type = 'sentence' AND content ? 'speaker_id'";
|
|
let sentence_chunks: Vec<String> = match sqlx::query_scalar(sentence_query)
|
|
.bind(&request.video_uuid)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
{
|
|
Ok(rows) => rows,
|
|
Err(_) => vec![],
|
|
};
|
|
|
|
// 4. Match speakers to persons by time overlap
|
|
let mut person_speaker_votes: HashMap<String, HashMap<String, f64>> = HashMap::new();
|
|
|
|
// Check ASRX trace chunks
|
|
for (_, content_text) in &asrx_chunks {
|
|
if let Ok(content) = serde_json::from_str::<serde_json::Value>(content_text) {
|
|
if let (Some(speaker_id), Some(start)) = (
|
|
content.get("speaker_id").and_then(|v| v.as_str()),
|
|
content.get("timestamp").and_then(|v| v.as_f64()),
|
|
) {
|
|
let end = start + 5.0; // Approximate 5s segments
|
|
for (pid, stat) in &person_stats {
|
|
for ts in &stat.timestamps {
|
|
if *ts >= start && *ts <= end {
|
|
person_speaker_votes
|
|
.entry(pid.clone())
|
|
.or_default()
|
|
.entry(speaker_id.to_string())
|
|
.and_modify(|v| *v += 1.0)
|
|
.or_insert(1.0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check sentence chunks for speaker_id
|
|
for content_text in &sentence_chunks {
|
|
if let Ok(content) = serde_json::from_str::<serde_json::Value>(content_text) {
|
|
if let (Some(_speaker_id), Some(_text)) = (
|
|
content.get("speaker_id").and_then(|v| v.as_str()),
|
|
content.get("text").and_then(|v| v.as_str()),
|
|
) {
|
|
// Timestamps not directly available in sentence chunks for ASRX matching
|
|
// Rely on ASRX trace chunks for precise matching
|
|
}
|
|
}
|
|
}
|
|
|
|
// 5. Insert/update person_identities
|
|
let min_conf = request.min_speaker_confidence.unwrap_or(0.0);
|
|
let mut matched_count = 0;
|
|
let mut persons_result = Vec::new();
|
|
|
|
// Sort by frame count descending
|
|
let mut sorted_persons: Vec<_> = person_stats.into_iter().collect();
|
|
sorted_persons.sort_by(|a, b| b.1.frame_count.cmp(&a.1.frame_count));
|
|
|
|
for (pid, stat) in sorted_persons {
|
|
let speaker_info = person_speaker_votes.get(&pid);
|
|
let (speaker_id, confidence) = if let Some(votes) = speaker_info {
|
|
let total: f64 = votes.values().sum();
|
|
if total > 0.0 {
|
|
let (best_speaker, best_votes) = votes
|
|
.iter()
|
|
.max_by(|a, b| a.1.partial_cmp(b.1).unwrap())
|
|
.unwrap();
|
|
let conf = best_votes / total;
|
|
if conf >= min_conf {
|
|
(Some(best_speaker.clone()), Some(conf))
|
|
} else {
|
|
(None, None)
|
|
}
|
|
} else {
|
|
(None, None)
|
|
}
|
|
} else {
|
|
(None, None)
|
|
};
|
|
|
|
if speaker_id.is_some() {
|
|
matched_count += 1;
|
|
}
|
|
|
|
// Upsert into person_identities
|
|
let upsert_query = r#"
|
|
INSERT INTO person_identities (person_id, name, speaker_id, first_appearance_time, last_appearance_time, appearance_count, metadata)
|
|
VALUES ($1, $2, $3, $4, $5, $6, $7)
|
|
ON CONFLICT (person_id) DO UPDATE SET
|
|
speaker_id = COALESCE(EXCLUDED.speaker_id, person_identities.speaker_id),
|
|
first_appearance_time = EXCLUDED.first_appearance_time,
|
|
last_appearance_time = EXCLUDED.last_appearance_time,
|
|
appearance_count = EXCLUDED.appearance_count,
|
|
metadata = COALESCE(EXCLUDED.metadata, person_identities.metadata),
|
|
updated_at = NOW()
|
|
RETURNING person_id, name, speaker_id, appearance_count, total_appearance_duration,
|
|
first_appearance_time, last_appearance_time, is_confirmed
|
|
"#;
|
|
|
|
let metadata = if let Some(conf) = confidence {
|
|
serde_json::json!({"auto_identified": true, "speaker_confidence": conf})
|
|
} else {
|
|
serde_json::json!({"auto_identified": true})
|
|
};
|
|
|
|
let result: Result<
|
|
Option<(
|
|
String,
|
|
Option<String>,
|
|
Option<String>,
|
|
i32,
|
|
f64,
|
|
Option<f64>,
|
|
Option<f64>,
|
|
bool,
|
|
)>,
|
|
_,
|
|
> = sqlx::query_as(upsert_query)
|
|
.bind(&pid)
|
|
.bind(&pid) // Use cluster label as initial name
|
|
.bind(&speaker_id)
|
|
.bind(stat.first_time)
|
|
.bind(stat.last_time)
|
|
.bind(stat.frame_count)
|
|
.bind(&metadata)
|
|
.fetch_optional(db.pool())
|
|
.await;
|
|
|
|
if let Ok(Some(row)) = result {
|
|
persons_result.push(PersonSummary {
|
|
person_id: row.0,
|
|
name: row.1,
|
|
speaker_id: row.2,
|
|
appearance_count: row.3,
|
|
total_appearance_duration: row.4,
|
|
first_appearance_time: row.5,
|
|
last_appearance_time: row.6,
|
|
is_confirmed: row.7,
|
|
speaker_confidence: confidence,
|
|
});
|
|
}
|
|
}
|
|
|
|
Ok(Json(AutoIdentifyResponse {
|
|
success: true,
|
|
message: format!(
|
|
"Identified {} persons, {} matched to speakers",
|
|
persons_result.len(),
|
|
matched_count
|
|
),
|
|
total_persons: persons_result.len() as i32,
|
|
matched_speakers: matched_count,
|
|
persons: persons_result,
|
|
}))
|
|
}
|
|
|
|
/// List all persons with optional filters
|
|
async fn list_persons(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Query(query): Query<PersonListQuery>,
|
|
) -> Result<Json<PersonListResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let limit = query.limit.unwrap_or(50) as i64;
|
|
let offset = query.offset.unwrap_or(0) as i64;
|
|
let min_appearances = query.min_appearances.unwrap_or(0);
|
|
let has_speaker = query.has_speaker.unwrap_or(false);
|
|
|
|
let (sql, count_sql) = if has_speaker {
|
|
if min_appearances > 0 {
|
|
(
|
|
"SELECT person_id, name, speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, is_confirmed, metadata::text FROM person_identities WHERE speaker_id IS NOT NULL AND appearance_count >= $1 ORDER BY appearance_count DESC LIMIT $2 OFFSET $3".to_string(),
|
|
"SELECT COUNT(*) FROM person_identities WHERE speaker_id IS NOT NULL AND appearance_count >= $1".to_string(),
|
|
)
|
|
} else {
|
|
(
|
|
"SELECT person_id, name, speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, is_confirmed, metadata::text FROM person_identities WHERE speaker_id IS NOT NULL ORDER BY appearance_count DESC LIMIT $1 OFFSET $2".to_string(),
|
|
"SELECT COUNT(*) FROM person_identities WHERE speaker_id IS NOT NULL".to_string(),
|
|
)
|
|
}
|
|
} else {
|
|
if min_appearances > 0 {
|
|
(
|
|
"SELECT person_id, name, speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, is_confirmed, metadata::text FROM person_identities WHERE appearance_count >= $1 ORDER BY appearance_count DESC LIMIT $2 OFFSET $3".to_string(),
|
|
"SELECT COUNT(*) FROM person_identities WHERE appearance_count >= $1".to_string(),
|
|
)
|
|
} else {
|
|
(
|
|
"SELECT person_id, name, speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, is_confirmed, metadata::text FROM person_identities ORDER BY appearance_count DESC LIMIT $1 OFFSET $2".to_string(),
|
|
"SELECT COUNT(*) FROM person_identities".to_string(),
|
|
)
|
|
}
|
|
};
|
|
|
|
let total: i64 = if min_appearances > 0 {
|
|
sqlx::query_scalar(&count_sql)
|
|
.bind(min_appearances)
|
|
.fetch_one(db.pool())
|
|
.await
|
|
.unwrap_or(0)
|
|
} else {
|
|
sqlx::query_scalar(&count_sql)
|
|
.fetch_one(db.pool())
|
|
.await
|
|
.unwrap_or(0)
|
|
};
|
|
|
|
let rows: Vec<(
|
|
String,
|
|
Option<String>,
|
|
Option<String>,
|
|
i32,
|
|
f64,
|
|
Option<f64>,
|
|
Option<f64>,
|
|
bool,
|
|
Option<String>,
|
|
)> = if has_speaker && min_appearances > 0 {
|
|
sqlx::query_as(&sql)
|
|
.bind(min_appearances)
|
|
.bind(limit)
|
|
.bind(offset)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
.unwrap_or_default()
|
|
} else if has_speaker {
|
|
sqlx::query_as(&sql)
|
|
.bind(limit)
|
|
.bind(offset)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
.unwrap_or_default()
|
|
} else if min_appearances > 0 {
|
|
sqlx::query_as(&sql)
|
|
.bind(min_appearances)
|
|
.bind(limit)
|
|
.bind(offset)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
.unwrap_or_default()
|
|
} else {
|
|
sqlx::query_as(&sql)
|
|
.bind(limit)
|
|
.bind(offset)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
.unwrap_or_default()
|
|
};
|
|
|
|
let persons: Vec<PersonSummary> = rows
|
|
.into_iter()
|
|
.map(|r| {
|
|
let speaker_confidence = r.8.as_ref().and_then(|m| {
|
|
serde_json::from_str::<serde_json::Value>(m)
|
|
.ok()
|
|
.and_then(|v| v.get("speaker_confidence").and_then(|v| v.as_f64()))
|
|
});
|
|
PersonSummary {
|
|
person_id: r.0,
|
|
name: r.1,
|
|
speaker_id: r.2,
|
|
appearance_count: r.3,
|
|
total_appearance_duration: r.4,
|
|
first_appearance_time: r.5,
|
|
last_appearance_time: r.6,
|
|
is_confirmed: r.7,
|
|
speaker_confidence,
|
|
}
|
|
})
|
|
.collect();
|
|
|
|
Ok(Json(PersonListResponse {
|
|
success: true,
|
|
persons,
|
|
total,
|
|
}))
|
|
}
|
|
|
|
/// Merge duplicate persons into one
|
|
async fn merge_persons(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Json(request): Json<MergePersonsRequest>,
|
|
) -> Result<Json<MergePersonsResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
if request.source_person_ids.is_empty() {
|
|
return Err((
|
|
StatusCode::BAD_REQUEST,
|
|
"source_person_ids cannot be empty".into(),
|
|
));
|
|
}
|
|
|
|
let mut tx = match db.pool().begin().await {
|
|
Ok(tx) => tx,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Tx error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// 0. Save original stats for undo capability
|
|
let orig_target_query = "SELECT appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time FROM person_identities WHERE person_id = $1";
|
|
let orig_target: Option<(i32, f64, Option<f64>, Option<f64>)> =
|
|
match sqlx::query_as(orig_target_query)
|
|
.bind(&request.target_person_id)
|
|
.fetch_optional(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Target query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
let orig_target = match orig_target {
|
|
Some(t) => t,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Target person not found: {}", request.target_person_id),
|
|
))
|
|
}
|
|
};
|
|
let orig_target_stats = serde_json::json!({
|
|
"appearance_count": orig_target.0,
|
|
"total_appearance_duration": orig_target.1,
|
|
"first_appearance_time": orig_target.2,
|
|
"last_appearance_time": orig_target.3,
|
|
});
|
|
|
|
let orig_sources_query = "SELECT person_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time FROM person_identities WHERE person_id = ANY($1)";
|
|
let orig_sources: Vec<(String, i32, f64, Option<f64>, Option<f64>)> =
|
|
match sqlx::query_as(orig_sources_query)
|
|
.bind(&request.source_person_ids)
|
|
.fetch_all(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Source query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
let orig_source_stats: Vec<serde_json::Value> = orig_sources
|
|
.into_iter()
|
|
.map(|(pid, count, dur, first, last)| {
|
|
serde_json::json!({
|
|
"person_id": pid,
|
|
"appearance_count": count,
|
|
"total_appearance_duration": dur,
|
|
"first_appearance_time": first,
|
|
"last_appearance_time": last,
|
|
})
|
|
})
|
|
.collect();
|
|
|
|
// Generate merge_id
|
|
let merge_id = Uuid::new_v4().to_string();
|
|
|
|
// A. Calculate sum of stats from sources
|
|
let stats_query = r#"
|
|
SELECT
|
|
COALESCE(SUM(appearance_count), 0)::integer as count,
|
|
COALESCE(SUM(total_appearance_duration), 0.0)::double precision as duration
|
|
FROM person_identities
|
|
WHERE person_id = ANY($1)
|
|
"#;
|
|
let (add_count, add_duration): (i32, f64) = match sqlx::query_as(stats_query)
|
|
.bind(&request.source_person_ids)
|
|
.fetch_one(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Stats error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// B. Transfer person_appearances
|
|
let move_query = "UPDATE person_appearances SET person_id = $1 WHERE person_id = ANY($2)";
|
|
match sqlx::query(move_query)
|
|
.bind(&request.target_person_id)
|
|
.bind(&request.source_person_ids)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Move error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// C. Delete source person_identities
|
|
let delete_query = "DELETE FROM person_identities WHERE person_id = ANY($1)";
|
|
match sqlx::query(delete_query)
|
|
.bind(&request.source_person_ids)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Delete error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// D. Update target stats
|
|
let update_query = r#"
|
|
UPDATE person_identities
|
|
SET
|
|
appearance_count = appearance_count + $1,
|
|
total_appearance_duration = total_appearance_duration + $2,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $3
|
|
"#;
|
|
match sqlx::query(update_query)
|
|
.bind(add_count)
|
|
.bind(add_duration)
|
|
.bind(&request.target_person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Update error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// E. Record merge history for undo capability
|
|
let history_query = r#"
|
|
INSERT INTO merge_history (merge_id, target_person_id, source_person_ids, original_target_stats, original_source_stats)
|
|
VALUES ($1::uuid, $2, $3, $4::jsonb, $5::jsonb)
|
|
"#;
|
|
let source_ids_json: Vec<String> = request.source_person_ids.clone();
|
|
let target_stats_json =
|
|
serde_json::to_string(&orig_target_stats).unwrap_or_else(|_| "{}".to_string());
|
|
let source_stats_json =
|
|
serde_json::to_string(&orig_source_stats).unwrap_or_else(|_| "[]".to_string());
|
|
match sqlx::query(history_query)
|
|
.bind(&merge_id)
|
|
.bind(&request.target_person_id)
|
|
.bind(&source_ids_json)
|
|
.bind(&target_stats_json)
|
|
.bind(&source_stats_json)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
tracing::warn!("[MERGE] Failed to record merge history: {}", e);
|
|
// Don't fail the merge if history recording fails
|
|
}
|
|
};
|
|
|
|
// F. Commit
|
|
match tx.commit().await {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Commit error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(MergePersonsResponse {
|
|
success: true,
|
|
message: format!(
|
|
"Merged {} persons into {}",
|
|
request.source_person_ids.len(),
|
|
request.target_person_id
|
|
),
|
|
target_person_id: request.target_person_id,
|
|
merge_id,
|
|
}))
|
|
}
|
|
|
|
/// Undo a previous merge
|
|
async fn undo_merge(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Json(request): Json<UndoMergeRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// 1. Get merge history
|
|
let history_query = "SELECT id, merge_id::text, target_person_id, source_person_ids, original_target_stats::text, original_source_stats::text, is_undone FROM merge_history WHERE merge_id = $1::uuid";
|
|
let history: Option<(i32, String, String, Vec<String>, String, String, bool)> =
|
|
match sqlx::query_as(history_query)
|
|
.bind(&request.merge_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("History query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let (history_id, _merge_id, target_id, source_ids, tgt_stats_str, src_stats_str, is_undone) =
|
|
match history {
|
|
Some(h) => h,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Merge history not found: {}", request.merge_id),
|
|
))
|
|
}
|
|
};
|
|
|
|
let orig_target_stats: serde_json::Value =
|
|
serde_json::from_str(&tgt_stats_str).unwrap_or(serde_json::json!({}));
|
|
let orig_source_stats: serde_json::Value =
|
|
serde_json::from_str(&src_stats_str).unwrap_or(serde_json::json!({}));
|
|
|
|
if is_undone {
|
|
return Err((
|
|
StatusCode::BAD_REQUEST,
|
|
"This merge has already been undone".into(),
|
|
));
|
|
}
|
|
|
|
let mut tx = match db.pool().begin().await {
|
|
Ok(tx) => tx,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Tx error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// 2. Restore target stats
|
|
let target_count = orig_target_stats
|
|
.get("appearance_count")
|
|
.and_then(|v| v.as_i64())
|
|
.unwrap_or(0) as i32;
|
|
let target_duration = orig_target_stats
|
|
.get("total_appearance_duration")
|
|
.and_then(|v| v.as_f64())
|
|
.unwrap_or(0.0);
|
|
let target_first = orig_target_stats
|
|
.get("first_appearance_time")
|
|
.and_then(|v| v.as_f64());
|
|
let target_last = orig_target_stats
|
|
.get("last_appearance_time")
|
|
.and_then(|v| v.as_f64());
|
|
|
|
let restore_target_query = "UPDATE person_identities SET appearance_count = $1, total_appearance_duration = $2, first_appearance_time = $3, last_appearance_time = $4, updated_at = CURRENT_TIMESTAMP WHERE person_id = $5";
|
|
match sqlx::query(restore_target_query)
|
|
.bind(target_count)
|
|
.bind(target_duration)
|
|
.bind(target_first)
|
|
.bind(target_last)
|
|
.bind(&target_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Restore target error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// 3. Recreate source person_identities
|
|
let empty_arr = vec![];
|
|
let source_stats_arr = orig_source_stats.as_array().unwrap_or(&empty_arr);
|
|
for source_stat in source_stats_arr {
|
|
let source_id = source_stat
|
|
.get("person_id")
|
|
.and_then(|v| v.as_str())
|
|
.unwrap_or("");
|
|
let source_count = source_stat
|
|
.get("appearance_count")
|
|
.and_then(|v| v.as_i64())
|
|
.unwrap_or(0) as i32;
|
|
let source_duration = source_stat
|
|
.get("total_appearance_duration")
|
|
.and_then(|v| v.as_f64())
|
|
.unwrap_or(0.0);
|
|
let source_first = source_stat
|
|
.get("first_appearance_time")
|
|
.and_then(|v| v.as_f64());
|
|
let source_last = source_stat
|
|
.get("last_appearance_time")
|
|
.and_then(|v| v.as_f64());
|
|
|
|
let restore_source_query = r#"
|
|
INSERT INTO person_identities (person_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, metadata, is_confirmed)
|
|
VALUES ($1, $2, $3, $4, $5, $6, FALSE)
|
|
ON CONFLICT (person_id) DO UPDATE SET
|
|
appearance_count = EXCLUDED.appearance_count,
|
|
total_appearance_duration = EXCLUDED.total_appearance_duration,
|
|
first_appearance_time = EXCLUDED.first_appearance_time,
|
|
last_appearance_time = EXCLUDED.last_appearance_time,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
"#;
|
|
match sqlx::query(restore_source_query)
|
|
.bind(source_id)
|
|
.bind(source_count)
|
|
.bind(source_duration)
|
|
.bind(source_first)
|
|
.bind(source_last)
|
|
.bind(&serde_json::json!({"restored_from_merge": _merge_id}))
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => tracing::warn!("[UNDO] Failed to restore {}: {}", source_id, e),
|
|
};
|
|
}
|
|
|
|
// 4. Mark merge history as undone
|
|
let mark_undone_query =
|
|
"UPDATE merge_history SET is_undone = TRUE, undone_at = CURRENT_TIMESTAMP WHERE id = $1";
|
|
match sqlx::query(mark_undone_query)
|
|
.bind(history_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Mark undone error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match tx.commit().await {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Commit error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Undo merge completed. Restored {} source persons", source_ids.len()),
|
|
"merge_id": _merge_id,
|
|
"target_person_id": target_id,
|
|
"restored_persons": source_ids
|
|
})))
|
|
}
|
|
|
|
/// Get merge history
|
|
async fn get_merge_history(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
) -> Result<Json<MergeHistoryResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let query = "SELECT merge_id::text, target_person_id, source_person_ids, original_target_stats::text, original_source_stats::text, merged_at, is_undone, undone_at FROM merge_history ORDER BY merged_at DESC LIMIT 50";
|
|
let rows: Vec<(
|
|
String,
|
|
String,
|
|
Vec<String>,
|
|
String,
|
|
String,
|
|
chrono::DateTime<chrono::Utc>,
|
|
bool,
|
|
Option<chrono::DateTime<chrono::Utc>>,
|
|
)> = match sqlx::query_as(query).fetch_all(db.pool()).await {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("History query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let history: Vec<MergeHistoryEntry> = rows
|
|
.into_iter()
|
|
.map(
|
|
|(merge_id, target, sources, tgt_stats, src_stats, merged_at, is_undone, undone_at)| {
|
|
MergeHistoryEntry {
|
|
merge_id,
|
|
target_person_id: target,
|
|
source_person_ids: sources,
|
|
original_target_stats: serde_json::from_str(&tgt_stats)
|
|
.unwrap_or(serde_json::json!({})),
|
|
original_source_stats: serde_json::from_str(&src_stats)
|
|
.unwrap_or(serde_json::json!({})),
|
|
merged_at: merged_at.to_rfc3339(),
|
|
is_undone,
|
|
undone_at: undone_at.map(|t| t.to_rfc3339()),
|
|
}
|
|
},
|
|
)
|
|
.collect();
|
|
|
|
Ok(Json(MergeHistoryResponse {
|
|
success: true,
|
|
history,
|
|
}))
|
|
}
|
|
|
|
/// Get AI suggestions for naming and merging persons
|
|
async fn get_person_suggestions(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Json(request): Json<AutoIdentifyRequest>,
|
|
) -> Result<Json<SuggestionsResponse>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Get all persons for this video (or all if no video_uuid)
|
|
let persons_query = if request.video_uuid.is_empty() {
|
|
"SELECT person_id, name, speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, metadata FROM person_identities ORDER BY appearance_count DESC LIMIT 100"
|
|
} else {
|
|
// Filter by video_uuid via person_appearances
|
|
"SELECT DISTINCT pi.person_id, pi.name, pi.speaker_id, pi.appearance_count, pi.total_appearance_duration, pi.first_appearance_time, pi.last_appearance_time, pi.metadata FROM person_identities pi JOIN person_appearances pa ON pi.person_id = pa.person_id WHERE pa.video_uuid = $1 ORDER BY pi.appearance_count DESC LIMIT 100"
|
|
};
|
|
|
|
let persons: Vec<(
|
|
String,
|
|
Option<String>,
|
|
Option<String>,
|
|
i32,
|
|
f64,
|
|
Option<f64>,
|
|
Option<f64>,
|
|
Option<serde_json::Value>,
|
|
)> = if request.video_uuid.is_empty() {
|
|
sqlx::query_as(persons_query)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
.unwrap_or_default()
|
|
} else {
|
|
sqlx::query_as(persons_query)
|
|
.bind(&request.video_uuid)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
.unwrap_or_default()
|
|
};
|
|
|
|
let mut naming_suggestions = Vec::new();
|
|
let mut merge_suggestions = Vec::new();
|
|
|
|
// Group by speaker_id to find merge candidates
|
|
let mut speaker_groups: std::collections::HashMap<String, Vec<(String, i32)>> =
|
|
std::collections::HashMap::new();
|
|
|
|
for (
|
|
person_id,
|
|
name,
|
|
speaker_id,
|
|
appearance_count,
|
|
_duration,
|
|
_first_time,
|
|
_last_time,
|
|
metadata,
|
|
) in &persons
|
|
{
|
|
// Naming suggestions
|
|
if name.is_none() {
|
|
let mut sources = Vec::new();
|
|
let mut suggested_name = person_id.clone();
|
|
let mut confidence = 0.0;
|
|
|
|
// Check speaker_id
|
|
if let Some(sid) = speaker_id {
|
|
sources.push(SuggestionSource {
|
|
r#type: "speaker_match".to_string(),
|
|
detail: format!("Linked to speaker {}", sid),
|
|
weight: 0.4,
|
|
});
|
|
suggested_name = sid.clone();
|
|
confidence += 0.4;
|
|
}
|
|
|
|
// Check metadata for speaker_confidence
|
|
if let Some(meta) = metadata {
|
|
if let Some(sc) = meta.get("speaker_confidence").and_then(|v| v.as_f64()) {
|
|
confidence += sc * 0.3;
|
|
sources.push(SuggestionSource {
|
|
r#type: "speaker_confidence".to_string(),
|
|
detail: format!("Speaker match confidence: {:.0}%", sc * 100.0),
|
|
weight: sc * 0.3,
|
|
});
|
|
}
|
|
}
|
|
|
|
// High appearance count suggests main character
|
|
if *appearance_count > 1000 {
|
|
confidence += 0.2;
|
|
sources.push(SuggestionSource {
|
|
r#type: "high_appearance".to_string(),
|
|
detail: format!(
|
|
"Appears in {} frames (likely main character)",
|
|
appearance_count
|
|
),
|
|
weight: 0.2,
|
|
});
|
|
}
|
|
|
|
if confidence > 0.0 {
|
|
let action = if confidence >= 0.7 {
|
|
"auto_apply".to_string()
|
|
} else {
|
|
"needs_review".to_string()
|
|
};
|
|
|
|
naming_suggestions.push(NamingSuggestion {
|
|
person_id: person_id.clone(),
|
|
current_name: name.clone(),
|
|
suggested_name,
|
|
confidence,
|
|
sources,
|
|
action,
|
|
});
|
|
}
|
|
}
|
|
|
|
// Group by speaker_id for merge detection
|
|
if let Some(sid) = speaker_id {
|
|
speaker_groups
|
|
.entry(sid.clone())
|
|
.or_default()
|
|
.push((person_id.clone(), *appearance_count));
|
|
}
|
|
}
|
|
|
|
// Find merge candidates: multiple persons with same speaker_id
|
|
for (speaker_id, group) in &speaker_groups {
|
|
if group.len() > 1 {
|
|
let primary = group.iter().max_by_key(|(_, count)| *count).unwrap();
|
|
let others: Vec<String> = group
|
|
.iter()
|
|
.filter(|(pid, _)| pid != &primary.0)
|
|
.map(|(pid, _)| pid.clone())
|
|
.collect();
|
|
|
|
// Calculate confidence based on appearance ratio
|
|
let primary_count = primary.1 as f64;
|
|
let total_count: f64 = group.iter().map(|(_, c)| *c as f64).sum();
|
|
let confidence = if total_count > 0.0 {
|
|
primary_count / total_count
|
|
} else {
|
|
0.0
|
|
};
|
|
|
|
let reasons = vec![
|
|
format!("All share speaker_id: {}", speaker_id),
|
|
format!(
|
|
"Primary {} has {} appearances ({}% of group)",
|
|
primary.0,
|
|
primary.1,
|
|
(primary_count / total_count * 100.0) as i32
|
|
),
|
|
format!("{} persons to merge into primary", others.len()),
|
|
];
|
|
|
|
let action = if confidence >= 0.7 {
|
|
"auto_apply".to_string()
|
|
} else {
|
|
"needs_review".to_string()
|
|
};
|
|
|
|
merge_suggestions.push(MergeSuggestion {
|
|
person_id: primary.0.clone(),
|
|
merge_with: others,
|
|
confidence,
|
|
reasons,
|
|
action,
|
|
});
|
|
}
|
|
}
|
|
|
|
let total_naming = naming_suggestions.len();
|
|
let total_merge = merge_suggestions.len();
|
|
|
|
Ok(Json(SuggestionsResponse {
|
|
success: true,
|
|
naming_suggestions,
|
|
merge_suggestions,
|
|
total_naming,
|
|
total_merge,
|
|
}))
|
|
}
|
|
|
|
/// Find similar persons that could be merged
|
|
async fn get_similar_persons(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Query(query): Query<SimilarPersonsQuery>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let threshold = query.threshold.unwrap_or(0.5);
|
|
let limit = query.limit.unwrap_or(10);
|
|
|
|
// Get the target person's speaker_id and time range
|
|
let person_query = "SELECT speaker_id, first_appearance_time, last_appearance_time FROM person_identities WHERE person_id = $1";
|
|
let person_info: Option<(Option<String>, Option<f64>, Option<f64>)> =
|
|
match sqlx::query_as(person_query)
|
|
.bind(&person_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(info) => info,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let person_info = match person_info {
|
|
Some(info) => info,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Find similar persons by speaker_id overlap or time overlap
|
|
let similar_query = r#"
|
|
SELECT
|
|
pi.person_id,
|
|
pi.name,
|
|
pi.speaker_id,
|
|
pi.appearance_count,
|
|
pi.first_appearance_time,
|
|
pi.last_appearance_time,
|
|
CASE
|
|
WHEN pi.speaker_id IS NOT NULL AND $2 IS NOT NULL AND pi.speaker_id = $2 THEN 0.7::double precision
|
|
WHEN pi.speaker_id IS NULL THEN 0.3::double precision
|
|
ELSE 0.5::double precision
|
|
END as similarity
|
|
FROM person_identities pi
|
|
WHERE pi.person_id != $1
|
|
AND pi.appearance_count > 0
|
|
ORDER BY similarity DESC, pi.appearance_count DESC
|
|
LIMIT $3
|
|
"#;
|
|
|
|
let similar: Vec<(
|
|
String,
|
|
Option<String>,
|
|
Option<String>,
|
|
i32,
|
|
Option<f64>,
|
|
Option<f64>,
|
|
f64,
|
|
)> = match sqlx::query_as(similar_query)
|
|
.bind(&person_id)
|
|
.bind(&person_info.0)
|
|
.bind(limit)
|
|
.fetch_all(db.pool())
|
|
.await
|
|
{
|
|
Ok(rows) => rows,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let results: Vec<serde_json::Value> = similar
|
|
.into_iter()
|
|
.filter(|(_, _, _, _, _, _, similarity)| *similarity >= threshold)
|
|
.map(|(pid, name, speaker_id, count, first, last, similarity)| {
|
|
serde_json::json!({
|
|
"person_id": pid,
|
|
"name": name,
|
|
"speaker_id": speaker_id,
|
|
"appearance_count": count,
|
|
"first_appearance_time": first,
|
|
"last_appearance_time": last,
|
|
"similarity": similarity
|
|
})
|
|
})
|
|
.collect();
|
|
|
|
Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"person_id": person_id,
|
|
"similar_persons": results,
|
|
"threshold": threshold
|
|
})))
|
|
}
|
|
|
|
/// Confirm an AI suggestion (auto-apply naming)
|
|
async fn confirm_person_suggestion(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<UpdatePersonIdentityRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let query = r#"
|
|
UPDATE person_identities
|
|
SET
|
|
name = COALESCE($2, name),
|
|
metadata = COALESCE($3, metadata),
|
|
is_confirmed = true,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $1
|
|
RETURNING person_id, name
|
|
"#;
|
|
|
|
let updated: Option<(String, Option<String>)> = match sqlx::query_as(query)
|
|
.bind(&person_id)
|
|
.bind(&request.name)
|
|
.bind(&request.metadata)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(result) => result,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Update error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match updated {
|
|
Some((id, name)) => Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Person '{}' confirmed", name.unwrap_or_else(|| id.clone())),
|
|
"person_id": id
|
|
}))),
|
|
None => Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
)),
|
|
}
|
|
}
|
|
|
|
// ============================================================
|
|
// Correction APIs - For fixing incorrect person bindings
|
|
// ============================================================
|
|
|
|
/// Request to unbind speaker from person
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct UnbindSpeakerRequest {
|
|
pub video_uuid: String,
|
|
pub reason: Option<String>,
|
|
}
|
|
|
|
/// Request to reassign speaker to person
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct ReassignSpeakerRequest {
|
|
pub video_uuid: String,
|
|
pub speaker_id: String,
|
|
pub reason: Option<String>,
|
|
}
|
|
|
|
/// Request to remove a specific appearance
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct RemoveAppearanceRequest {
|
|
pub video_uuid: String,
|
|
pub appearance_id: i32,
|
|
pub reason: Option<String>,
|
|
}
|
|
|
|
/// Request to reassign appearance to another person
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct ReassignAppearanceRequest {
|
|
pub video_uuid: String,
|
|
pub appearance_id: i32,
|
|
pub target_person_id: String,
|
|
pub reason: Option<String>,
|
|
}
|
|
|
|
/// Request to split a person into two
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct SplitPersonRequest {
|
|
pub video_uuid: String,
|
|
pub new_person_id: String,
|
|
pub appearance_ids_to_move: Vec<i32>,
|
|
pub new_person_name: Option<String>,
|
|
pub reason: Option<String>,
|
|
}
|
|
|
|
/// Unbind speaker from person
|
|
async fn unbind_speaker(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<UnbindSpeakerRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let update_query = r#"
|
|
UPDATE person_identities
|
|
SET speaker_id = NULL, updated_at = CURRENT_TIMESTAMP,
|
|
metadata = jsonb_set(
|
|
COALESCE(metadata, '{}'::jsonb),
|
|
'{speaker_unbound}',
|
|
'true'::jsonb
|
|
)
|
|
WHERE person_id = $1
|
|
RETURNING person_id
|
|
"#;
|
|
|
|
let updated: Option<String> = match sqlx::query_scalar(update_query)
|
|
.bind(&person_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Unbind error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match updated {
|
|
Some(id) => Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Speaker unbound from person '{}'", id),
|
|
"person_id": id,
|
|
"reason": request.reason.unwrap_or_default()
|
|
}))),
|
|
None => Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
)),
|
|
}
|
|
}
|
|
|
|
/// Reassign speaker to person
|
|
async fn reassign_speaker(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<ReassignSpeakerRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let update_query = r#"
|
|
UPDATE person_identities
|
|
SET speaker_id = $2, updated_at = CURRENT_TIMESTAMP,
|
|
metadata = jsonb_set(
|
|
COALESCE(metadata, '{}'::jsonb),
|
|
'{speaker_reassigned}',
|
|
'true'::jsonb
|
|
)
|
|
WHERE person_id = $1
|
|
RETURNING person_id
|
|
"#;
|
|
|
|
let updated: Option<String> = match sqlx::query_scalar(update_query)
|
|
.bind(&person_id)
|
|
.bind(&request.speaker_id)
|
|
.fetch_optional(db.pool())
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Reassign error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match updated {
|
|
Some(id) => Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Speaker '{}' assigned to person '{}'", request.speaker_id, id),
|
|
"person_id": id,
|
|
"new_speaker_id": request.speaker_id,
|
|
"reason": request.reason.unwrap_or_default()
|
|
}))),
|
|
None => Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
)),
|
|
}
|
|
}
|
|
|
|
/// Remove a specific appearance
|
|
async fn remove_appearance(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<RemoveAppearanceRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let mut tx = match db.pool().begin().await {
|
|
Ok(tx) => tx,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Tx error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Get appearance info before deleting
|
|
let app_query =
|
|
"SELECT duration, person_id FROM person_appearances WHERE id = $1 AND person_id = $2";
|
|
let app_info: Option<(f64, String)> = match sqlx::query_as(app_query)
|
|
.bind(request.appearance_id)
|
|
.bind(&person_id)
|
|
.fetch_optional(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Appearance query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let (duration, actual_person_id) = match app_info {
|
|
Some(info) => info,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Appearance not found: {}", request.appearance_id),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Delete appearance
|
|
let delete_query = "DELETE FROM person_appearances WHERE id = $1 AND person_id = $2";
|
|
match sqlx::query(delete_query)
|
|
.bind(request.appearance_id)
|
|
.bind(&person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Delete error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Update person stats
|
|
let update_stats_query = r#"
|
|
UPDATE person_identities
|
|
SET appearance_count = appearance_count - 1,
|
|
total_appearance_duration = GREATEST(0, total_appearance_duration - $1),
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $2
|
|
"#;
|
|
match sqlx::query(update_stats_query)
|
|
.bind(duration)
|
|
.bind(&actual_person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Stats update error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match tx.commit().await {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Commit error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Appearance {} removed from person '{}'", request.appearance_id, person_id),
|
|
"appearance_id": request.appearance_id,
|
|
"person_id": person_id,
|
|
"removed_duration": duration,
|
|
"reason": request.reason.unwrap_or_default()
|
|
})))
|
|
}
|
|
|
|
/// Reassign appearance to another person
|
|
async fn reassign_appearance(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<ReassignAppearanceRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let mut tx = match db.pool().begin().await {
|
|
Ok(tx) => tx,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Tx error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Get appearance info
|
|
let app_query =
|
|
"SELECT id, duration, person_id FROM person_appearances WHERE id = $1 AND person_id = $2";
|
|
let app_info: Option<(i32, f64, String)> = match sqlx::query_as(app_query)
|
|
.bind(request.appearance_id)
|
|
.bind(&person_id)
|
|
.fetch_optional(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Appearance query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let (app_id, duration, _old_person_id) = match app_info {
|
|
Some(info) => info,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Appearance not found: {}", request.appearance_id),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Reassign to new person
|
|
let update_query = "UPDATE person_appearances SET person_id = $1 WHERE id = $2";
|
|
match sqlx::query(update_query)
|
|
.bind(&request.target_person_id)
|
|
.bind(app_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Reassign error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Update old person stats (decrement)
|
|
let update_old_query = r#"
|
|
UPDATE person_identities
|
|
SET appearance_count = GREATEST(0, appearance_count - 1),
|
|
total_appearance_duration = GREATEST(0, total_appearance_duration - $1),
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $2
|
|
"#;
|
|
match sqlx::query(update_old_query)
|
|
.bind(duration)
|
|
.bind(&person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Old person stats error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Update new person stats (increment)
|
|
let update_new_query = r#"
|
|
UPDATE person_identities
|
|
SET appearance_count = appearance_count + 1,
|
|
total_appearance_duration = total_appearance_duration + $1,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $2
|
|
"#;
|
|
match sqlx::query(update_new_query)
|
|
.bind(duration)
|
|
.bind(&request.target_person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("New person stats error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
match tx.commit().await {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Commit error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Appearance {} reassigned from '{}' to '{}'", request.appearance_id, person_id, request.target_person_id),
|
|
"appearance_id": request.appearance_id,
|
|
"from_person_id": person_id,
|
|
"to_person_id": request.target_person_id,
|
|
"reason": request.reason.unwrap_or_default()
|
|
})))
|
|
}
|
|
|
|
/// Split a person into two
|
|
async fn split_person(
|
|
State(_state): State<crate::api::server::AppState>,
|
|
Path(person_id): Path<String>,
|
|
Json(request): Json<SplitPersonRequest>,
|
|
) -> Result<Json<serde_json::Value>, (StatusCode, String)> {
|
|
let db = match PostgresDb::init().await {
|
|
Ok(db) => db,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("DB error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let mut tx = match db.pool().begin().await {
|
|
Ok(tx) => tx,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Tx error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Get original person info
|
|
let orig_query = "SELECT speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, metadata FROM person_identities WHERE person_id = $1";
|
|
let orig_info: Option<(
|
|
Option<String>,
|
|
i32,
|
|
f64,
|
|
Option<f64>,
|
|
Option<f64>,
|
|
Option<serde_json::Value>,
|
|
)> = match sqlx::query_as(orig_query)
|
|
.bind(&person_id)
|
|
.fetch_optional(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Person query error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
let (speaker_id, orig_count, orig_duration, orig_first, orig_last, orig_metadata) =
|
|
match orig_info {
|
|
Some(info) => info,
|
|
None => {
|
|
return Err((
|
|
StatusCode::NOT_FOUND,
|
|
format!("Person not found: {}", person_id),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Create new person
|
|
let new_count = request.appearance_ids_to_move.len();
|
|
let create_query = r#"
|
|
INSERT INTO person_identities (person_id, name, speaker_id, appearance_count, total_appearance_duration, first_appearance_time, last_appearance_time, metadata, is_confirmed)
|
|
VALUES ($1, $2, $3, $4, 0, NULL, NULL, $5, FALSE)
|
|
ON CONFLICT (person_id) DO UPDATE SET
|
|
name = EXCLUDED.name,
|
|
speaker_id = EXCLUDED.speaker_id,
|
|
metadata = EXCLUDED.metadata
|
|
RETURNING person_id
|
|
"#;
|
|
let new_name = request
|
|
.new_person_name
|
|
.unwrap_or_else(|| format!("{}-split", request.new_person_id));
|
|
let mut new_metadata = orig_metadata.unwrap_or(serde_json::json!({}));
|
|
new_metadata["split_from"] = serde_json::json!(person_id);
|
|
|
|
match sqlx::query(create_query)
|
|
.bind(&request.new_person_id)
|
|
.bind(new_name)
|
|
.bind(&speaker_id)
|
|
.bind(new_count as i32)
|
|
.bind(&new_metadata)
|
|
.fetch_optional(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Create person error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Move appearances to new person and calculate new stats
|
|
let mut new_duration: f64 = 0.0;
|
|
let mut new_first: Option<f64> = None;
|
|
let mut new_last: Option<f64> = None;
|
|
|
|
for app_id in &request.appearance_ids_to_move {
|
|
let app_query = "SELECT duration, start_time, end_time FROM person_appearances WHERE id = $1 AND person_id = $2";
|
|
let app_info: Option<(f64, f64, f64)> = match sqlx::query_as(app_query)
|
|
.bind(app_id)
|
|
.bind(&person_id)
|
|
.fetch_optional(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
tracing::warn!("[SPLIT] Failed to get appearance {}: {}", app_id, e);
|
|
continue;
|
|
}
|
|
};
|
|
|
|
if let Some((dur, start, end)) = app_info {
|
|
// Update appearance to new person
|
|
let update_app_query = "UPDATE person_appearances SET person_id = $1 WHERE id = $2";
|
|
match sqlx::query(update_app_query)
|
|
.bind(&request.new_person_id)
|
|
.bind(app_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
tracing::warn!("[SPLIT] Failed to update appearance {}: {}", app_id, e);
|
|
continue;
|
|
}
|
|
};
|
|
|
|
new_duration += dur;
|
|
if new_first.is_none() || Some(start) < new_first {
|
|
new_first = Some(start);
|
|
}
|
|
if new_last.is_none() || Some(end) > new_last {
|
|
new_last = Some(end);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Update new person stats
|
|
let update_new_query = r#"
|
|
UPDATE person_identities
|
|
SET total_appearance_duration = $1,
|
|
first_appearance_time = $2,
|
|
last_appearance_time = $3,
|
|
appearance_count = $4,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $5
|
|
"#;
|
|
match sqlx::query(update_new_query)
|
|
.bind(new_duration)
|
|
.bind(new_first)
|
|
.bind(new_last)
|
|
.bind(new_count as i32)
|
|
.bind(&request.new_person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Update new person error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Update original person stats (decrement)
|
|
let update_orig_query = r#"
|
|
UPDATE person_identities
|
|
SET appearance_count = GREATEST(0, appearance_count - $1),
|
|
total_appearance_duration = GREATEST(0, total_appearance_duration - $2),
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE person_id = $3
|
|
"#;
|
|
match sqlx::query(update_orig_query)
|
|
.bind(new_count as i32)
|
|
.bind(new_duration)
|
|
.bind(&person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Update original person error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
// Update original person's first/last appearance times
|
|
let recalc_query = r#"
|
|
UPDATE person_identities
|
|
SET first_appearance_time = (SELECT MIN(start_time) FROM person_appearances WHERE person_id = $1),
|
|
last_appearance_time = (SELECT MAX(end_time) FROM person_appearances WHERE person_id = $1)
|
|
WHERE person_id = $1
|
|
"#;
|
|
match sqlx::query(recalc_query)
|
|
.bind(&person_id)
|
|
.execute(&mut *tx)
|
|
.await
|
|
{
|
|
Ok(_) => {}
|
|
Err(e) => tracing::warn!("[SPLIT] Failed to recalc original person times: {}", e),
|
|
};
|
|
|
|
match tx.commit().await {
|
|
Ok(_) => {}
|
|
Err(e) => {
|
|
return Err((
|
|
StatusCode::INTERNAL_SERVER_ERROR,
|
|
format!("Commit error: {}", e),
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(Json(serde_json::json!({
|
|
"success": true,
|
|
"message": format!("Person '{}' split into '{}' with {} appearances moved", person_id, request.new_person_id, new_count),
|
|
"original_person_id": person_id,
|
|
"new_person_id": request.new_person_id,
|
|
"appearances_moved": new_count,
|
|
"new_person_duration": new_duration,
|
|
"new_person_first": new_first,
|
|
"new_person_last": new_last,
|
|
"reason": request.reason.unwrap_or_default()
|
|
})))
|
|
}
|