upend/src/routes.rs

192 lines
6.2 KiB
Rust
Raw Normal View History

use crate::addressing::Address;
2021-02-19 21:58:35 +01:00
use crate::database::{
2021-02-21 19:51:23 +01:00
get_latest_files, insert_entry, query, remove_object, retrieve_file, retrieve_object, DbPool,
Entry, Query,
2021-02-19 21:58:35 +01:00
};
use crate::filesystem::{list_directory, UPath};
2021-02-21 17:08:33 +01:00
use crate::hash::{decode, encode};
use crate::jobs::JobContainer;
2020-08-27 01:07:25 +02:00
use actix_files::NamedFile;
use actix_web::error::{ErrorBadRequest, ErrorInternalServerError, ErrorNotFound};
2021-02-19 21:58:35 +01:00
use actix_web::{delete, error, get, post, put, web, Error, HttpResponse};
use anyhow::Result;
2021-02-19 21:58:35 +01:00
use futures_util::StreamExt;
use log::debug;
use serde::Deserialize;
use std::collections::HashMap;
use std::convert::TryFrom;
use std::path::PathBuf;
use std::sync::{Arc, RwLock};
#[derive(Clone)]
2020-08-27 01:07:25 +02:00
pub struct State {
pub directory: PathBuf,
pub db_pool: DbPool,
pub job_container: Arc<RwLock<JobContainer>>,
2020-08-27 01:07:25 +02:00
}
#[get("/api/raw/{hash}")]
2020-08-27 01:07:25 +02:00
pub async fn get_raw(state: web::Data<State>, hash: web::Path<String>) -> Result<NamedFile, Error> {
let address = Address::decode(&decode(hash.into_inner()).map_err(ErrorInternalServerError)?)
.map_err(ErrorInternalServerError)?;
if let Address::Hash(hash) = address {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
let response = retrieve_file(&connection, hash);
2020-08-27 01:07:25 +02:00
debug!("{:?}", response);
match response {
Ok(result) => match result {
Some(path) => Ok(NamedFile::open(state.directory.join(path))?),
None => Err(error::ErrorNotFound("NOT FOUND")),
},
Err(e) => Err(error::ErrorInternalServerError(e)),
}
} else {
Err(ErrorBadRequest("Address does not refer to a file."))
2020-08-27 01:07:25 +02:00
}
}
#[derive(Deserialize)]
pub struct QueryRequest {
query: String,
}
#[get("/api/obj")]
pub async fn get_query(
state: web::Data<State>,
web::Query(info): web::Query<QueryRequest>,
) -> Result<HttpResponse, Error> {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
2021-02-21 10:38:31 +01:00
let sexp = lexpr::from_str(info.query.as_str()).map_err(ErrorBadRequest)?;
let in_query = Query::try_from(&sexp).map_err(ErrorBadRequest)?;
2021-02-21 12:30:17 +01:00
let entries = query(&connection, in_query).map_err(ErrorInternalServerError)?;
let mut result: HashMap<String, Entry> = HashMap::new();
for entry in entries {
result.insert(
2021-02-21 17:08:33 +01:00
encode(
entry
.address()
.map_err(ErrorInternalServerError)?
.encode()
.map_err(ErrorInternalServerError)?,
),
2021-02-21 12:30:17 +01:00
entry,
);
}
2021-02-21 12:30:17 +01:00
Ok(HttpResponse::Ok().json(&result))
}
#[get("/api/obj/{address_str}")]
pub async fn get_object(
state: web::Data<State>,
address_str: web::Path<String>,
) -> Result<HttpResponse, Error> {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
let response: Result<Vec<Entry>> = retrieve_object(
&connection,
Address::decode(&decode(address_str.into_inner()).map_err(ErrorBadRequest)?)
2021-02-21 17:08:33 +01:00
.map_err(ErrorBadRequest)?,
);
debug!("{:?}", response);
let mut result: HashMap<String, Entry> = HashMap::new();
for entry in response.map_err(error::ErrorInternalServerError)? {
result.insert(
2021-02-21 17:08:33 +01:00
encode(
entry
.address()
.map_err(ErrorInternalServerError)?
.encode()
.map_err(ErrorInternalServerError)?,
),
entry,
);
}
Ok(HttpResponse::Ok().json(result))
}
2021-02-19 21:58:35 +01:00
const MAX_SIZE: usize = 1_000_000;
#[put("/api/obj")]
pub async fn put_object(
state: web::Data<State>,
mut payload: web::Payload,
) -> Result<HttpResponse, Error> {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
let mut body = web::BytesMut::new();
while let Some(chunk) = payload.next().await {
let chunk = chunk?;
// limit max size of in-memory payload
if (body.len() + chunk.len()) > MAX_SIZE {
return Err(error::ErrorBadRequest("overflow."));
}
body.extend_from_slice(&chunk);
}
2021-02-21 17:49:34 +01:00
let entry = serde_json::from_slice::<Entry>(&body).map_err(ErrorBadRequest)?;
2021-02-19 21:58:35 +01:00
let result_address = insert_entry(&connection, entry).map_err(ErrorInternalServerError)?;
2021-02-19 21:58:35 +01:00
Ok(HttpResponse::Ok().json(result_address))
2021-02-19 21:58:35 +01:00
}
#[delete("/api/obj/{address_str}")]
pub async fn delete_object(
state: web::Data<State>,
address_str: web::Path<String>,
) -> Result<HttpResponse, Error> {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
let _ = remove_object(
&connection,
Address::decode(&decode(address_str.into_inner()).map_err(ErrorBadRequest)?)
.map_err(ErrorInternalServerError)?,
)
.map_err(ErrorInternalServerError)?;
Ok(HttpResponse::Ok().finish())
}
#[get("/api/hier/{path:.*}")]
pub async fn list_hier(
state: web::Data<State>,
path: web::Path<String>,
) -> Result<HttpResponse, Error> {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
let upath: UPath = path.into_inner().parse().map_err(ErrorBadRequest)?;
let entries: Vec<Entry> = list_directory(&connection, &upath)
.await
.map_err(ErrorNotFound)?; // todo: 500 if actual error occurs
Ok(HttpResponse::Ok().json(entries))
}
2020-08-27 01:07:25 +02:00
#[post("/api/refresh")]
pub async fn api_refresh(state: web::Data<State>) -> Result<HttpResponse, Error> {
2020-09-20 19:28:44 +02:00
let _pool = state.db_pool.clone();
let _directory = state.directory.clone();
actix::spawn(crate::filesystem::reimport_directory(
_pool,
_directory,
state.job_container.clone(),
));
2020-08-27 01:07:25 +02:00
Ok(HttpResponse::Ok().finish())
}
2021-02-21 19:51:23 +01:00
#[get("/api/files/latest")]
pub async fn latest_files(state: web::Data<State>) -> Result<HttpResponse, Error> {
let connection = state.db_pool.get().map_err(ErrorInternalServerError)?;
let files = get_latest_files(&connection, 100).map_err(ErrorInternalServerError)?;
Ok(HttpResponse::Ok().json(&files))
}
#[get("/api/jobs")]
pub async fn get_jobs(state: web::Data<State>) -> Result<HttpResponse, Error> {
let jobs = state.job_container.read().unwrap().get_jobs();
Ok(HttpResponse::Ok().json(&jobs))
}