use actix_web::{
HttpRequest, HttpResponse, delete,
http::StatusCode,
web::{Data, Json},
};
use serde::Deserialize;
use serde_json::json;
use std::time::Instant;
use supabase_rs::SupabaseClient;
use crate::AppState;
use crate::api::gateway::auth::{authorize_gateway_request, delete_right_for_resource};
use crate::api::headers::x_athena_client::x_athena_client;
use crate::api::headers::x_company_id::get_x_company_id;
use crate::api::headers::x_organization_id::get_x_organization_id;
use crate::api::headers::x_publish_event::get_x_publish_event;
use crate::api::headers::x_user_id::get_x_user_id;
use crate::data::events::post_event;
use crate::drivers::supabase::client_router;
use crate::utils::request_logging::{LoggedRequest, log_operation_event, log_request};
#[derive(Debug, Deserialize)]
struct DeleteRequest {
table_name: String,
resource_id: String,
}
#[delete("/gateway/delete")]
pub async fn delete_data(
req: HttpRequest,
body: Json<DeleteRequest>,
app_state: Data<AppState>,
) -> HttpResponse {
let operation_start: Instant = Instant::now();
let client_name: String = x_athena_client(&req.clone());
let auth = authorize_gateway_request(
&req,
app_state.get_ref(),
Some(&client_name),
vec![delete_right_for_resource(Some(&body.table_name))],
)
.await;
let logged_request: LoggedRequest = log_request(
req.clone(),
Some(app_state.get_ref()),
Some(auth.request_id.clone()),
Some(&auth.log_context),
);
if let Some(resp) = auth.response {
return resp;
}
#[allow(unused)]
let _user_id: Option<String> = get_x_user_id(&req);
let _company_id: String = match get_x_company_id(&req) {
Some(id) => id,
None => {
return HttpResponse::BadRequest()
.json(json!({"error": "X-Company-Id header not found in the request"}));
}
};
let _organization_id: String = match get_x_organization_id(&req) {
Some(id) => id,
None => {
return HttpResponse::BadRequest()
.json(json!({"error": "X-Organization-Id header not found in the request"}));
}
};
let table_name: String = body.table_name.clone();
let resource_id: String = body.resource_id.clone();
let client: SupabaseClient = match client_router(&client_name).await {
Ok(c) => c,
Err(err) => {
return HttpResponse::BadRequest().json(json!({
"error": err,
"message": "Failed to resolve Supabase client",
}));
}
};
let result: Result<(), String> = client.delete(&table_name, &resource_id).await;
match result {
Ok(_) => {
app_state.cache.invalidate_all();
if get_x_publish_event(&req) {
let event: serde_json::Value = json!({
"event": "DELETE",
"resource": table_name.clone(),
"resource_id": resource_id.clone(),
});
post_event(_company_id, event).await;
}
log_operation_event(
Some(app_state.get_ref()),
&logged_request,
"delete",
Some(&table_name),
operation_start.elapsed().as_millis(),
StatusCode::OK,
Some(json!({
"resource_id": resource_id.clone(),
})),
);
HttpResponse::Ok().json(json!({
"status": "success",
"success": true,
"message": "Data deleted successfully",
"table_name": table_name,
"resource_id": resource_id,
"client": client_name,
}))
}
Err(err) => {
log_operation_event(
Some(app_state.get_ref()),
&logged_request,
"delete",
Some(&table_name),
operation_start.elapsed().as_millis(),
StatusCode::INTERNAL_SERVER_ERROR,
Some(json!({
"error": err,
})),
);
HttpResponse::InternalServerError().json(json!({
"status": "error",
"success": false,
"message": format!("Failed to delete: {:?}", err),
"table_name": table_name,
"resource_id": resource_id,
}))
}
}
}