rieter/server/src/repo/mod.rs

118 lines
3.3 KiB
Rust
Raw Normal View History

mod manager;
2023-07-12 21:51:07 +00:00
mod package;
pub use manager::RepoGroupManager;
2023-07-12 09:04:31 +00:00
use axum::extract::{BodyStream, Path, State};
use axum::http::StatusCode;
use axum::routing::{delete, get_service, post};
2023-07-11 11:41:56 +00:00
use axum::Router;
2023-07-12 09:04:31 +00:00
use futures::StreamExt;
use std::sync::Arc;
use tokio::{fs, io::AsyncWriteExt};
2023-07-11 11:41:56 +00:00
use tower_http::services::ServeDir;
2023-07-12 09:04:31 +00:00
use uuid::Uuid;
2023-07-11 11:41:56 +00:00
pub fn router(global: &crate::Global) -> Router<crate::Global> {
2023-07-11 11:41:56 +00:00
// Try to serve packages by default, and try the database files instead if not found
let serve_repos = get_service(
ServeDir::new(&global.config.pkg_dir).fallback(ServeDir::new(&global.config.repo_dir)),
);
2023-07-11 11:41:56 +00:00
Router::new()
2023-07-12 09:04:31 +00:00
.route(
"/:repo",
post(post_package_archive)
.delete(delete_repo)
.get(serve_repos.clone()),
2023-07-12 09:04:31 +00:00
)
.route("/:repo/:arch", delete(delete_arch_repo))
.route("/:repo/:arch/:filename", delete(delete_package))
2023-07-12 09:04:31 +00:00
.fallback(serve_repos)
.with_state(global.clone())
2023-07-11 11:41:56 +00:00
}
2023-07-12 09:04:31 +00:00
async fn post_package_archive(
State(global): State<crate::Global>,
2023-07-12 09:04:31 +00:00
Path(repo): Path<String>,
mut body: BodyStream,
) -> crate::Result<()> {
2023-07-12 09:04:31 +00:00
// We first stream the uploaded file to disk
let uuid: uuid::fmt::Simple = Uuid::new_v4().into();
let path = global.config.pkg_dir.join(uuid.to_string());
let mut f = fs::File::create(&path).await?;
2023-07-12 09:04:31 +00:00
while let Some(chunk) = body.next().await {
f.write_all(&chunk?).await?;
2023-07-12 09:04:31 +00:00
}
let clone = Arc::clone(&global.repo_manager);
tokio::task::spawn_blocking(move || clone.write().unwrap().add_pkg_from_path(&repo, &path))
.await??;
Ok(())
2023-07-12 09:04:31 +00:00
}
async fn delete_repo(
State(global): State<crate::Global>,
Path(repo): Path<String>,
) -> crate::Result<StatusCode> {
let clone = Arc::clone(&global.repo_manager);
let repo_removed =
tokio::task::spawn_blocking(move || clone.write().unwrap().remove_repo(&repo)).await??;
if repo_removed {
Ok(StatusCode::OK)
} else {
Ok(StatusCode::NOT_FOUND)
}
}
async fn delete_arch_repo(
State(global): State<crate::Global>,
Path((repo, arch)): Path<(String, String)>,
) -> crate::Result<StatusCode> {
let clone = Arc::clone(&global.repo_manager);
let repo_removed =
tokio::task::spawn_blocking(move || clone.write().unwrap().remove_arch_repo(&repo, &arch))
.await??;
if repo_removed {
Ok(StatusCode::OK)
} else {
Ok(StatusCode::NOT_FOUND)
}
}
async fn delete_package(
State(global): State<crate::Global>,
Path((repo, arch, file_name)): Path<(String, String, String)>,
) -> crate::Result<StatusCode> {
let name_parts = file_name.split('-').collect::<Vec<_>>();
// Package archive files use the naming scheme pkgname-pkgver-pkgrel-arch, so a valid
// name contains at least 4 dash-separated sections
if name_parts.len() < 4 {
return Ok(StatusCode::NOT_FOUND);
}
let name = name_parts[..name_parts.len() - 3].join("-");
let clone = Arc::clone(&global.repo_manager);
let pkg_removed = tokio::task::spawn_blocking(move || {
clone
.write()
.unwrap()
.remove_pkg_from_arch_repo(&repo, &arch, &name, true)
})
.await??;
if pkg_removed {
Ok(StatusCode::OK)
} else {
Ok(StatusCode::NOT_FOUND)
}
}