use axum::extract::{Json, State};
use axum::response::{IntoResponse, Response};
use common::prelude::MountError;
use reqwest::{Client, RequestBuilder, Url};
use serde::{Deserialize, Serialize};
use uuid::Uuid;
use common::crypto::PublicKey;
use crate::http_server::api::client::ApiRequest;
use crate::ServiceState;
#[derive(Debug, Clone, Copy, Default, Serialize, Deserialize, PartialEq, Eq, clap::ValueEnum)]
#[serde(rename_all = "lowercase")]
pub enum ShareRole {
#[default]
Owner,
Mirror,
}
#[derive(Debug, Clone, Serialize, Deserialize, clap::Args)]
pub struct ShareRequest {
#[arg(long)]
pub bucket_id: Uuid,
#[arg(long)]
pub peer_public_key: String,
#[arg(long, default_value = "owner")]
#[serde(default)]
pub role: ShareRole,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ShareResponse {
pub bucket_id: Uuid,
pub peer_public_key: String,
pub new_bucket_link: String,
}
pub async fn handler(
State(state): State<ServiceState>,
Json(req): Json<ShareRequest>,
) -> Result<impl IntoResponse, ShareError> {
tracing::info!(
"SHARE API: Received share request for bucket {} with peer {} as {:?}",
req.bucket_id,
req.peer_public_key,
req.role
);
let peer_public_key = PublicKey::from_hex(&req.peer_public_key)
.map_err(|e| ShareError::InvalidPublicKey(e.to_string()))?;
tracing::info!("SHARE API: Parsed peer public key successfully");
let mut mount = state.peer().mount(req.bucket_id).await?;
tracing::info!("SHARE API: Loaded mount for bucket {}", req.bucket_id);
match req.role {
ShareRole::Owner => {
mount.add_owner(peer_public_key).await?;
tracing::info!(
"SHARE API: Mount.add_owner() completed for peer {}",
req.peer_public_key
);
}
ShareRole::Mirror => {
mount.add_mirror(peer_public_key).await;
tracing::info!(
"SHARE API: Mount.add_mirror() completed for peer {}",
req.peer_public_key
);
}
}
tracing::info!("SHARE API: Calling save_mount for bucket {}", req.bucket_id);
let new_bucket_link = state.peer().save_mount(&mount, None).await?;
tracing::info!(
"SHARE API: Bucket {} shared with peer {} as {:?}, new link: {}",
req.bucket_id,
req.peer_public_key,
req.role,
new_bucket_link.hash()
);
Ok((
http::StatusCode::OK,
Json(ShareResponse {
bucket_id: req.bucket_id,
peer_public_key: req.peer_public_key,
new_bucket_link: new_bucket_link.hash().to_string(),
}),
)
.into_response())
}
#[derive(Debug, thiserror::Error)]
pub enum ShareError {
#[error("Invalid public key: {0}")]
InvalidPublicKey(String),
#[error("Mount error: {0}")]
Mount(#[from] MountError),
}
impl IntoResponse for ShareError {
fn into_response(self) -> Response {
match self {
ShareError::InvalidPublicKey(msg) => (
http::StatusCode::BAD_REQUEST,
format!("Invalid public key: {}", msg),
)
.into_response(),
ShareError::Mount(_) => (
http::StatusCode::INTERNAL_SERVER_ERROR,
"Unexpected error".to_string(),
)
.into_response(),
}
}
}
impl ApiRequest for ShareRequest {
type Response = ShareResponse;
fn build_request(self, base_url: &Url, client: &Client) -> RequestBuilder {
let full_url = base_url.join("/api/v0/bucket/share").unwrap();
client.post(full_url).json(&self)
}
}