use axum::extract::{Json, State};
use axum::response::{IntoResponse, Response};
use reqwest::{Client, RequestBuilder, Url};
use serde::{Deserialize, Serialize};
use uuid::Uuid;
use common::mount::Manifest;
use common::peer::sync::{DownloadPinsJob, SyncJob};
use crate::database::types::BucketStatus;
use crate::http_server::api::client::ApiRequest;
use crate::ServiceState;
#[derive(Debug, Clone, Serialize, Deserialize, clap::Args)]
pub struct ApproveRequest {
#[arg(long)]
pub bucket_id: Uuid,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ApproveResponse {
pub bucket_id: Uuid,
pub status: String,
}
pub async fn handler(
State(state): State<ServiceState>,
Json(req): Json<ApproveRequest>,
) -> Result<impl IntoResponse, ApproveError> {
tracing::info!("APPROVE BUCKET: {}", req.bucket_id);
let current_status = state
.database()
.get_effective_bucket_status(&req.bucket_id)
.await
.map_err(|e| ApproveError::Database(e.to_string()))?;
if current_status == BucketStatus::Active {
return Ok((
http::StatusCode::OK,
Json(ApproveResponse {
bucket_id: req.bucket_id,
status: "active".to_string(),
}),
)
.into_response());
}
state
.database()
.set_bucket_status(&req.bucket_id, BucketStatus::Active, None)
.await
.map_err(|e| ApproveError::Database(e.to_string()))?;
let logs = state
.database()
.get_all_bucket_logs(&req.bucket_id)
.await
.map_err(|e| ApproveError::Database(e.to_string()))?;
for entry in logs {
match state
.peer()
.blobs()
.get_cbor::<Manifest>(&entry.current_link.hash())
.await
{
Ok(manifest) => {
let pins_link = manifest.pins().clone();
let peer_ids = manifest.get_peer_ids();
if let Err(e) = state
.peer()
.dispatch(SyncJob::DownloadPins(DownloadPinsJob {
pins_link,
peer_ids,
}))
.await
{
tracing::warn!(
"Failed to dispatch catch-up pin download for height {}: {}",
entry.height,
e
);
}
}
Err(e) => {
tracing::warn!(
"Failed to load manifest at height {} for catch-up: {}",
entry.height,
e
);
}
}
}
tracing::info!(
"APPROVE BUCKET: {} approved and catch-up triggered",
req.bucket_id
);
Ok((
http::StatusCode::OK,
Json(ApproveResponse {
bucket_id: req.bucket_id,
status: "active".to_string(),
}),
)
.into_response())
}
#[derive(Debug, thiserror::Error)]
pub enum ApproveError {
#[error("Database error: {0}")]
Database(String),
}
impl IntoResponse for ApproveError {
fn into_response(self) -> Response {
(http::StatusCode::INTERNAL_SERVER_ERROR, format!("{}", self)).into_response()
}
}
impl ApiRequest for ApproveRequest {
type Response = ApproveResponse;
fn build_request(self, base_url: &Url, client: &Client) -> RequestBuilder {
let full_url = base_url.join("/api/v0/bucket/approve").unwrap();
client.post(full_url).json(&self)
}
}