use std::fmt::Debug;
use http::header::CONTENT_LENGTH;
use http::header::CONTENT_TYPE;
use http::HeaderValue;
use http::Request;
use http::Response;
use crate::raw::*;
use crate::*;
pub struct SupabaseCore {
pub root: String,
pub bucket: String,
pub endpoint: String,
pub key: Option<String>,
pub http_client: HttpClient,
}
impl Debug for SupabaseCore {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("SupabaseCore")
.field("root", &self.root)
.field("bucket", &self.bucket)
.field("endpoint", &self.endpoint)
.finish_non_exhaustive()
}
}
impl SupabaseCore {
pub fn new(
root: &str,
bucket: &str,
endpoint: &str,
key: Option<String>,
client: HttpClient,
) -> Self {
Self {
root: root.to_string(),
bucket: bucket.to_string(),
endpoint: endpoint.to_string(),
key,
http_client: client,
}
}
pub fn sign<T>(&self, req: &mut Request<T>) -> Result<()> {
if let Some(k) = &self.key {
let v = HeaderValue::from_str(&format!("Bearer {}", k)).unwrap();
req.headers_mut().insert(http::header::AUTHORIZATION, v);
}
Ok(())
}
}
impl SupabaseCore {
pub fn supabase_upload_object_request(
&self,
path: &str,
size: Option<usize>,
content_type: Option<&str>,
body: Buffer,
) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
let mut req = Request::post(&url);
if let Some(size) = size {
req = req.header(CONTENT_LENGTH, size)
}
if let Some(mime) = content_type {
req = req.header(CONTENT_TYPE, mime)
}
let req = req.body(body).map_err(new_request_build_error)?;
Ok(req)
}
pub fn supabase_get_object_public_request(
&self,
path: &str,
_: BytesRange,
) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/public/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
let req = Request::get(&url);
req.body(Buffer::new()).map_err(new_request_build_error)
}
pub fn supabase_get_object_auth_request(
&self,
path: &str,
_: BytesRange,
) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/authenticated/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
let req = Request::get(&url);
req.body(Buffer::new()).map_err(new_request_build_error)
}
pub fn supabase_head_object_public_request(&self, path: &str) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/public/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
Request::head(&url)
.body(Buffer::new())
.map_err(new_request_build_error)
}
pub fn supabase_head_object_auth_request(&self, path: &str) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/authenticated/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
Request::head(&url)
.body(Buffer::new())
.map_err(new_request_build_error)
}
pub fn supabase_get_object_info_public_request(&self, path: &str) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/info/public/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
Request::get(&url)
.body(Buffer::new())
.map_err(new_request_build_error)
}
pub fn supabase_get_object_info_auth_request(&self, path: &str) -> Result<Request<Buffer>> {
let p = build_abs_path(&self.root, path);
let url = format!(
"{}/storage/v1/object/info/authenticated/{}/{}",
self.endpoint,
self.bucket,
percent_encode_path(&p)
);
Request::get(&url)
.body(Buffer::new())
.map_err(new_request_build_error)
}
}
impl SupabaseCore {
pub async fn send(&self, req: Request<Buffer>) -> Result<Response<Buffer>> {
self.http_client.send(req).await
}
pub async fn supabase_get_object(
&self,
path: &str,
range: BytesRange,
) -> Result<Response<HttpBody>> {
let mut req = if self.key.is_some() {
self.supabase_get_object_auth_request(path, range)?
} else {
self.supabase_get_object_public_request(path, range)?
};
self.sign(&mut req)?;
self.http_client.fetch(req).await
}
pub async fn supabase_head_object(&self, path: &str) -> Result<Response<Buffer>> {
let mut req = if self.key.is_some() {
self.supabase_head_object_auth_request(path)?
} else {
self.supabase_head_object_public_request(path)?
};
self.sign(&mut req)?;
self.send(req).await
}
pub async fn supabase_get_object_info(&self, path: &str) -> Result<Response<Buffer>> {
let mut req = if self.key.is_some() {
self.supabase_get_object_info_auth_request(path)?
} else {
self.supabase_get_object_info_public_request(path)?
};
self.sign(&mut req)?;
self.send(req).await
}
}