|
use std::collections::HashMap; |
|
use std::io::Write; |
|
use actix_multipart_extract::{File, Multipart, MultipartForm}; |
|
use actix_web::{get, HttpResponse, post, web}; |
|
use chrono::{Utc, FixedOffset}; |
|
use minio::s3::args::{BucketExistsArgs, MakeBucketArgs, UploadObjectArgs}; |
|
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; |
|
|
|
const BUCKET_NAME: &'static str = "docgpt-upload"; |
|
|
|
|
|
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, |
|
} |
|
|
|
#[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 s3_client = &data.s3_client; |
|
let buckets_exists = s3_client |
|
.bucket_exists(&BucketExistsArgs::new(BUCKET_NAME)?) |
|
.await?; |
|
if !buckets_exists { |
|
s3_client |
|
.make_bucket(&MakeBucketArgs::new(BUCKET_NAME)?) |
|
.await?; |
|
} |
|
|
|
s3_client |
|
.upload_object( |
|
&mut UploadObjectArgs::new( |
|
BUCKET_NAME, |
|
fnm.as_str(), |
|
format!("/{}/{}-{}", payload.uid, payload.did, fnm).as_str() |
|
)? |
|
) |
|
.await?; |
|
|
|
let location = format!("/{}/{}", BUCKET_NAME, fnm); |
|
let doc = Mutation::create_doc_info(&data.conn, Model { |
|
did:Default::default(), |
|
uid: uid, |
|
doc_name: fnm, |
|
size: payload.file_field.bytes.len() as i64, |
|
location, |
|
r#type: "doc".to_string(), |
|
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(), |
|
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)?) |
|
) |
|
} |
|
|