AppFlowy-Cloud/libs/client-api/src/http_blob.rs

160 lines
4.6 KiB
Rust

use crate::http::log_request_id;
use crate::Client;
use app_error::AppError;
use bytes::Bytes;
use futures_util::StreamExt;
use mime::Mime;
use reqwest::{header, Method, StatusCode};
use shared_entity::dto::workspace_dto::{BlobMetadata, RepeatedBlobMetaData};
use shared_entity::response::{AppResponse, AppResponseError};
use tracing::instrument;
impl Client {
pub fn get_blob_url(&self, workspace_id: &str, file_id: &str) -> String {
format!(
"{}/api/file_storage/{}/blob/{}",
self.base_url, workspace_id, file_id
)
}
#[instrument(level = "info", skip_all)]
pub async fn put_blob<T: Into<Bytes>>(
&self,
url: &str,
data: T,
mime: &Mime,
) -> Result<(), AppResponseError> {
let data = data.into();
let resp = self
.http_client_with_auth(Method::PUT, url)
.await?
.header(header::CONTENT_TYPE, mime.to_string())
.body(data)
.send()
.await?;
log_request_id(&resp);
if resp.status() == StatusCode::PAYLOAD_TOO_LARGE {
return Err(AppResponseError::from(AppError::PayloadTooLarge(
StatusCode::PAYLOAD_TOO_LARGE.to_string(),
)));
}
AppResponse::<()>::from_response(resp).await?.into_error()
}
/// Only expose this method for testing
#[instrument(level = "info", skip_all)]
#[cfg(debug_assertions)]
pub async fn put_blob_with_content_length<T: Into<Bytes>>(
&self,
url: &str,
data: T,
mime: &Mime,
content_length: usize,
) -> Result<crate::entity::AFBlobRecord, AppResponseError> {
let resp = self
.http_client_with_auth(Method::PUT, url)
.await?
.header(header::CONTENT_TYPE, mime.to_string())
.header(header::CONTENT_LENGTH, content_length)
.body(data.into())
.send()
.await?;
log_request_id(&resp);
AppResponse::<crate::entity::AFBlobRecord>::from_response(resp)
.await?
.into_data()
}
/// Get the file with the given url. The url should be in the format of
/// `https://appflowy.io/api/file_storage/<workspace_id>/<file_id>`.
#[instrument(level = "info", skip_all)]
pub async fn get_blob(&self, url: &str) -> Result<(Mime, Vec<u8>), AppResponseError> {
let resp = self
.http_client_with_auth(Method::GET, url)
.await?
.send()
.await?;
log_request_id(&resp);
match resp.status() {
reqwest::StatusCode::OK => {
// get mime from resp header
let mime = {
match resp.headers().get(header::CONTENT_TYPE) {
Some(v) => match v.to_str() {
Ok(v) => match v.parse::<Mime>() {
Ok(v) => v,
Err(e) => {
tracing::error!("failed to parse mime from header: {:?}", e);
mime::TEXT_PLAIN
},
},
Err(e) => {
tracing::error!("failed to get mime from header: {:?}", e);
mime::TEXT_PLAIN
},
},
None => mime::TEXT_PLAIN,
}
};
let mut stream = resp.bytes_stream();
let mut acc: Vec<u8> = Vec::new();
while let Some(raw_bytes) = stream.next().await {
acc.extend_from_slice(&raw_bytes?);
}
Ok((mime, acc))
},
reqwest::StatusCode::NOT_FOUND => Err(AppResponseError::from(AppError::RecordNotFound(
url.to_owned(),
))),
c => Err(AppResponseError::from(AppError::Unhandled(format!(
"status code: {}, message: {}",
c,
resp.text().await?
)))),
}
}
#[instrument(level = "info", skip_all)]
pub async fn get_blob_metadata(&self, url: &str) -> Result<BlobMetadata, AppResponseError> {
let resp = self
.http_client_with_auth(Method::GET, url)
.await?
.send()
.await?;
log_request_id(&resp);
AppResponse::<BlobMetadata>::from_response(resp)
.await?
.into_data()
}
#[instrument(level = "info", skip_all)]
pub async fn delete_blob(&self, url: &str) -> Result<(), AppResponseError> {
let resp = self
.http_client_with_auth(Method::DELETE, url)
.await?
.send()
.await?;
log_request_id(&resp);
AppResponse::<()>::from_response(resp).await?.into_error()
}
pub async fn get_workspace_all_blob_metadata(
&self,
workspace_id: &str,
) -> Result<RepeatedBlobMetaData, AppResponseError> {
let url = format!("{}/api/file_storage/{}/blobs", self.base_url, workspace_id);
let resp = self
.http_client_with_auth(Method::GET, &url)
.await?
.send()
.await?;
log_request_id(&resp);
AppResponse::<RepeatedBlobMetaData>::from_response(resp)
.await?
.into_data()
}
}