|
use std::collections::HashMap; |
|
use std::io::BufReader; |
|
use actix_multipart_extract::{ File, Multipart, MultipartForm }; |
|
use actix_web::{ HttpResponse, post, web }; |
|
use chrono::{ Utc, FixedOffset }; |
|
use minio::s3::args::{ BucketExistsArgs, MakeBucketArgs, PutObjectArgs }; |
|
use sea_orm::DbConn; |
|
use crate::api::JsonResponse; |
|
use crate::AppState; |
|
use crate::entity::doc_info::Model; |
|
use crate::errors::AppError; |
|
use crate::service::doc_info::{ Mutation, Query }; |
|
use serde::Deserialize; |
|
use regex::Regex; |
|
|
|
fn now() -> chrono::DateTime<FixedOffset> { |
|
Utc::now().with_timezone(&FixedOffset::east_opt(3600 * 8).unwrap()) |
|
} |
|
|
|
#[derive(Debug, Deserialize)] |
|
pub struct ListParams { |
|
pub uid: i64, |
|
pub filter: FilterParams, |
|
pub sortby: String, |
|
pub page: Option<u32>, |
|
pub per_page: Option<u32>, |
|
} |
|
|
|
#[derive(Debug, Deserialize)] |
|
pub struct FilterParams { |
|
pub keywords: Option<String>, |
|
pub folder_id: Option<i64>, |
|
pub tag_id: Option<i64>, |
|
pub kb_id: Option<i64>, |
|
} |
|
|
|
#[post("/v1.0/docs")] |
|
async fn list( |
|
params: web::Json<ListParams>, |
|
data: web::Data<AppState> |
|
) -> Result<HttpResponse, AppError> { |
|
let docs = Query::find_doc_infos_by_params(&data.conn, params.into_inner()).await?; |
|
|
|
let mut result = HashMap::new(); |
|
result.insert("docs", docs); |
|
|
|
let json_response = JsonResponse { |
|
code: 200, |
|
err: "".to_owned(), |
|
data: result, |
|
}; |
|
|
|
Ok( |
|
HttpResponse::Ok() |
|
.content_type("application/json") |
|
.body(serde_json::to_string(&json_response)?) |
|
) |
|
} |
|
|
|
#[derive(Deserialize, MultipartForm, Debug)] |
|
pub struct UploadForm { |
|
#[multipart(max_size = 512MB)] |
|
file_field: File, |
|
uid: i64, |
|
did: i64, |
|
} |
|
|
|
fn file_type(filename: &String) -> String { |
|
let fnm = filename.to_lowercase(); |
|
if |
|
let Some(_) = Regex::new(r"\.(mpg|mpeg|avi|rm|rmvb|mov|wmv|asf|dat|asx|wvx|mpe|mpa)$") |
|
.unwrap() |
|
.captures(&fnm) |
|
{ |
|
return "Video".to_owned(); |
|
} |
|
if |
|
let Some(_) = Regex::new( |
|
r"\.(jpg|jpeg|png|tif|gif|pcx|tga|exif|fpx|svg|psd|cdr|pcd|dxf|ufo|eps|ai|raw|WMF|webp|avif|apng)$" |
|
) |
|
.unwrap() |
|
.captures(&fnm) |
|
{ |
|
return "Picture".to_owned(); |
|
} |
|
if |
|
let Some(_) = Regex::new(r"\.(WAV|FLAC|APE|ALAC|WavPack|WV|MP3|AAC|Ogg|Vorbis|Opus)$") |
|
.unwrap() |
|
.captures(&fnm) |
|
{ |
|
return "Music".to_owned(); |
|
} |
|
if |
|
let Some(_) = Regex::new(r"\.(pdf|doc|ppt|yml|xml|htm|json|csv|txt|ini|xsl|wps|rtf|hlp)$") |
|
.unwrap() |
|
.captures(&fnm) |
|
{ |
|
return "Document".to_owned(); |
|
} |
|
"Other".to_owned() |
|
} |
|
|
|
#[post("/v1.0/upload")] |
|
async fn upload( |
|
payload: Multipart<UploadForm>, |
|
data: web::Data<AppState> |
|
) -> Result<HttpResponse, AppError> { |
|
let uid = payload.uid; |
|
let file_name = payload.file_field.name.as_str(); |
|
async fn add_number_to_filename( |
|
file_name: &str, |
|
conn: &DbConn, |
|
uid: i64, |
|
parent_id: i64 |
|
) -> String { |
|
let mut i = 0; |
|
let mut new_file_name = file_name.to_string(); |
|
let arr: Vec<&str> = file_name.split(".").collect(); |
|
let suffix = String::from(arr[arr.len() - 1]); |
|
let preffix = arr[..arr.len() - 1].join("."); |
|
let mut docs = Query::find_doc_infos_by_name( |
|
conn, |
|
uid, |
|
&new_file_name, |
|
Some(parent_id) |
|
).await.unwrap(); |
|
while docs.len() > 0 { |
|
i += 1; |
|
new_file_name = format!("{}_{}.{}", preffix, i, suffix); |
|
docs = Query::find_doc_infos_by_name( |
|
conn, |
|
uid, |
|
&new_file_name, |
|
Some(parent_id) |
|
).await.unwrap(); |
|
} |
|
new_file_name |
|
} |
|
let fnm = add_number_to_filename(file_name, &data.conn, uid, payload.did).await; |
|
|
|
let bucket_name = format!("{}-upload", payload.uid); |
|
let s3_client: &minio::s3::client::Client = &data.s3_client; |
|
let buckets_exists = s3_client |
|
.bucket_exists(&BucketExistsArgs::new(&bucket_name).unwrap()).await |
|
.unwrap(); |
|
if !buckets_exists { |
|
print!("Create bucket: {}", bucket_name.clone()); |
|
s3_client.make_bucket(&MakeBucketArgs::new(&bucket_name).unwrap()).await.unwrap(); |
|
} else { |
|
print!("Existing bucket: {}", bucket_name.clone()); |
|
} |
|
|
|
let location = format!("/{}/{}", payload.did, fnm) |
|
.as_bytes() |
|
.to_vec() |
|
.iter() |
|
.map(|b| format!("{:02x}", b).to_string()) |
|
.collect::<Vec<String>>() |
|
.join(""); |
|
print!("===>{}", location.clone()); |
|
s3_client.put_object( |
|
&mut PutObjectArgs::new( |
|
&bucket_name, |
|
&location, |
|
&mut BufReader::new(payload.file_field.bytes.as_slice()), |
|
Some(payload.file_field.bytes.len()), |
|
None |
|
)? |
|
).await?; |
|
|
|
let doc = Mutation::create_doc_info(&data.conn, Model { |
|
did: Default::default(), |
|
uid: uid, |
|
doc_name: fnm.clone(), |
|
size: payload.file_field.bytes.len() as i64, |
|
location, |
|
r#type: file_type(&fnm), |
|
thumbnail_base64: Default::default(), |
|
created_at: now(), |
|
updated_at: now(), |
|
is_deleted: Default::default(), |
|
}).await?; |
|
|
|
let _ = Mutation::place_doc(&data.conn, payload.did, doc.did.unwrap()).await?; |
|
|
|
Ok(HttpResponse::Ok().body("File uploaded successfully")) |
|
} |
|
|
|
#[derive(Deserialize, Debug)] |
|
pub struct RmDocsParam { |
|
uid: i64, |
|
dids: Vec<i64>, |
|
} |
|
#[post("/v1.0/delete_docs")] |
|
async fn delete( |
|
params: web::Json<RmDocsParam>, |
|
data: web::Data<AppState> |
|
) -> Result<HttpResponse, AppError> { |
|
let _ = Mutation::delete_doc_info(&data.conn, ¶ms.dids).await?; |
|
|
|
let json_response = JsonResponse { |
|
code: 200, |
|
err: "".to_owned(), |
|
data: (), |
|
}; |
|
|
|
Ok( |
|
HttpResponse::Ok() |
|
.content_type("application/json") |
|
.body(serde_json::to_string(&json_response)?) |
|
) |
|
} |
|
|
|
#[derive(Debug, Deserialize)] |
|
pub struct MvParams { |
|
pub uid: i64, |
|
pub dids: Vec<i64>, |
|
pub dest_did: i64, |
|
} |
|
|
|
#[post("/v1.0/mv_docs")] |
|
async fn mv( |
|
params: web::Json<MvParams>, |
|
data: web::Data<AppState> |
|
) -> Result<HttpResponse, AppError> { |
|
Mutation::mv_doc_info(&data.conn, params.dest_did, ¶ms.dids).await?; |
|
|
|
let json_response = JsonResponse { |
|
code: 200, |
|
err: "".to_owned(), |
|
data: (), |
|
}; |
|
|
|
Ok( |
|
HttpResponse::Ok() |
|
.content_type("application/json") |
|
.body(serde_json::to_string(&json_response)?) |
|
) |
|
} |
|
|
|
#[derive(Debug, Deserialize)] |
|
pub struct NewFoldParams { |
|
pub uid: i64, |
|
pub parent_id: i64, |
|
pub name: String, |
|
} |
|
|
|
#[post("/v1.0/new_folder")] |
|
async fn new_folder( |
|
params: web::Json<NewFoldParams>, |
|
data: web::Data<AppState> |
|
) -> Result<HttpResponse, AppError> { |
|
let doc = Mutation::create_doc_info(&data.conn, Model { |
|
did: Default::default(), |
|
uid: params.uid, |
|
doc_name: params.name.to_string(), |
|
size: 0, |
|
r#type: "folder".to_string(), |
|
location: "".to_owned(), |
|
thumbnail_base64: Default::default(), |
|
created_at: now(), |
|
updated_at: now(), |
|
is_deleted: Default::default(), |
|
}).await?; |
|
let _ = Mutation::place_doc(&data.conn, params.parent_id, doc.did.unwrap()).await?; |
|
|
|
Ok(HttpResponse::Ok().body("Folder created successfully")) |
|
} |
|
|
|
#[derive(Debug, Deserialize)] |
|
pub struct RenameParams { |
|
pub uid: i64, |
|
pub did: i64, |
|
pub name: String, |
|
} |
|
|
|
#[post("/v1.0/rename")] |
|
async fn rename( |
|
params: web::Json<RenameParams>, |
|
data: web::Data<AppState> |
|
) -> Result<HttpResponse, AppError> { |
|
let docs = Query::find_doc_infos_by_name(&data.conn, params.uid, ¶ms.name, None).await?; |
|
if docs.len() > 0 { |
|
let json_response = JsonResponse { |
|
code: 500, |
|
err: "Name duplicated!".to_owned(), |
|
data: (), |
|
}; |
|
return Ok( |
|
HttpResponse::Ok() |
|
.content_type("application/json") |
|
.body(serde_json::to_string(&json_response)?) |
|
); |
|
} |
|
let doc = Mutation::rename(&data.conn, params.did, ¶ms.name).await?; |
|
|
|
let json_response = JsonResponse { |
|
code: 200, |
|
err: "".to_owned(), |
|
data: doc, |
|
}; |
|
|
|
Ok( |
|
HttpResponse::Ok() |
|
.content_type("application/json") |
|
.body(serde_json::to_string(&json_response)?) |
|
) |
|
} |
|
|