feat(server): working archive upload for non-any architecture
parent
eb9f6a5cb1
commit
c2f35aebac
libarchive/src
server/src/repo
|
@ -68,7 +68,7 @@ impl ReadFilter {
|
||||||
ReadFilter::Bzip2 => Some(".bz2"),
|
ReadFilter::Bzip2 => Some(".bz2"),
|
||||||
ReadFilter::Lzma => Some(".lzma"),
|
ReadFilter::Lzma => Some(".lzma"),
|
||||||
ReadFilter::Xz => Some(".xz"),
|
ReadFilter::Xz => Some(".xz"),
|
||||||
ReadFilter::Zstd => Some(".zstd"),
|
ReadFilter::Zstd => Some(".zst"),
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -308,6 +308,12 @@ pub trait Entry {
|
||||||
ffi::archive_entry_set_pathname(self.entry_mut(), c_str.as_ptr());
|
ffi::archive_entry_set_pathname(self.entry_mut(), c_str.as_ptr());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn set_mode(&mut self, mode: u32) {
|
||||||
|
unsafe {
|
||||||
|
ffi::archive_entry_set_mode(self.entry_mut(), mode);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq)]
|
#[derive(Debug, PartialEq, Eq)]
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
pub mod archive;
|
pub mod archive;
|
||||||
pub mod error;
|
pub mod error;
|
||||||
pub mod read;
|
pub mod read;
|
||||||
mod write;
|
pub mod write;
|
||||||
|
|
||||||
pub use archive::{
|
pub use archive::{
|
||||||
Entry, ExtractOption, ExtractOptions, Handle, ReadCompression, ReadFilter, ReadFormat,
|
Entry, ExtractOption, ExtractOptions, Handle, ReadCompression, ReadFilter, ReadFormat,
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
use super::WriteEntry;
|
use super::WriteEntry;
|
||||||
|
use crate::error::ArchiveError;
|
||||||
use crate::Entry;
|
use crate::Entry;
|
||||||
use crate::Handle;
|
use crate::Handle;
|
||||||
use libarchive3_sys::ffi;
|
use libarchive3_sys::ffi;
|
||||||
use libarchive3_sys::ffi::c_void;
|
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
|
@ -37,7 +37,10 @@ impl FileWriter {
|
||||||
path: P,
|
path: P,
|
||||||
) -> io::Result<()> {
|
) -> io::Result<()> {
|
||||||
unsafe {
|
unsafe {
|
||||||
ffi::archive_write_header(self.handle_mut(), entry.entry_mut());
|
match ffi::archive_write_header(self.handle_mut(), entry.entry_mut()) {
|
||||||
|
ffi::ARCHIVE_OK => (),
|
||||||
|
_ => return Err(ArchiveError::from(self as &dyn Handle).into()),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut f = fs::File::open(path)?;
|
let mut f = fs::File::open(path)?;
|
||||||
|
@ -47,7 +50,15 @@ impl FileWriter {
|
||||||
match f.read(&mut buf) {
|
match f.read(&mut buf) {
|
||||||
Ok(0) => return Ok(()),
|
Ok(0) => return Ok(()),
|
||||||
Ok(written) => unsafe {
|
Ok(written) => unsafe {
|
||||||
ffi::archive_write_data(self.handle_mut(), buf.as_ptr() as *const _, written);
|
match ffi::archive_write_data(
|
||||||
|
self.handle_mut(),
|
||||||
|
buf.as_ptr() as *const _,
|
||||||
|
written,
|
||||||
|
) as i32
|
||||||
|
{
|
||||||
|
ffi::ARCHIVE_OK => (),
|
||||||
|
_ => return Err(ArchiveError::from(self as &dyn Handle).into()),
|
||||||
|
};
|
||||||
},
|
},
|
||||||
Err(err) => match err.kind() {
|
Err(err) => match err.kind() {
|
||||||
io::ErrorKind::Interrupted => (),
|
io::ErrorKind::Interrupted => (),
|
||||||
|
|
|
@ -1,3 +1,8 @@
|
||||||
|
use super::package::Package;
|
||||||
|
use libarchive::write::{Builder, WriteEntry};
|
||||||
|
use libarchive::{Entry, WriteFilter, WriteFormat};
|
||||||
|
use std::fs;
|
||||||
|
use std::io;
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
/// Overarching abstraction that orchestrates updating the repositories stored on the server
|
/// Overarching abstraction that orchestrates updating the repositories stored on the server
|
||||||
|
@ -13,4 +18,160 @@ impl RepoGroupManager {
|
||||||
pkg_dir: pkg_dir.as_ref().to_path_buf(),
|
pkg_dir: pkg_dir.as_ref().to_path_buf(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn sync(&mut self, repo: &str, arch: &str) -> io::Result<()> {
|
||||||
|
let subrepo_path = self.repo_dir.join(repo).join(arch);
|
||||||
|
|
||||||
|
let mut ar_db = Builder::new();
|
||||||
|
ar_db.add_filter(WriteFilter::Gzip)?;
|
||||||
|
ar_db.set_format(WriteFormat::PaxRestricted)?;
|
||||||
|
|
||||||
|
let mut ar_files = Builder::new();
|
||||||
|
ar_files.add_filter(WriteFilter::Gzip)?;
|
||||||
|
ar_files.set_format(WriteFormat::PaxRestricted)?;
|
||||||
|
|
||||||
|
let mut ar_db = ar_db.open_file(subrepo_path.join(format!("{}.tar.gz", repo)))?;
|
||||||
|
let mut ar_files =
|
||||||
|
ar_files.open_file(subrepo_path.join(format!("{}.files.tar.gz", repo)))?;
|
||||||
|
|
||||||
|
for entry in subrepo_path.read_dir()? {
|
||||||
|
let entry = entry?;
|
||||||
|
|
||||||
|
if entry.file_type()?.is_dir() {
|
||||||
|
// The desc file needs to be added to both archives
|
||||||
|
let path_in_tar = PathBuf::from(entry.file_name()).join("desc");
|
||||||
|
let src_path = entry.path().join("desc");
|
||||||
|
|
||||||
|
let mut ar_entry = WriteEntry::new();
|
||||||
|
ar_entry.set_pathname(&path_in_tar);
|
||||||
|
ar_entry.set_mode(0o100644);
|
||||||
|
|
||||||
|
ar_db.append_path(&mut ar_entry, &src_path)?;
|
||||||
|
ar_files.append_path(&mut ar_entry, src_path)?;
|
||||||
|
|
||||||
|
// The files file is only required in the files database
|
||||||
|
let path_in_tar = PathBuf::from(entry.file_name()).join("files");
|
||||||
|
let src_path = entry.path().join("files");
|
||||||
|
|
||||||
|
let mut ar_entry = WriteEntry::new();
|
||||||
|
ar_entry.set_pathname(&path_in_tar);
|
||||||
|
ar_entry.set_mode(0o100644);
|
||||||
|
|
||||||
|
ar_files.append_path(&mut ar_entry, src_path)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn remove_pkg_from_arch_repo(
|
||||||
|
&mut self,
|
||||||
|
repo: &str,
|
||||||
|
arch: &str,
|
||||||
|
pkg_name: &str,
|
||||||
|
) -> io::Result<bool> {
|
||||||
|
let arch_repo_dir = self.repo_dir.join(repo).join(arch);
|
||||||
|
|
||||||
|
if !arch_repo_dir.exists() {
|
||||||
|
return Ok(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
for entry in arch_repo_dir.read_dir()? {
|
||||||
|
let entry = entry?;
|
||||||
|
|
||||||
|
// Make sure we skip the archive files
|
||||||
|
if !entry.metadata()?.is_dir() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let file_name = entry.file_name();
|
||||||
|
let file_name = file_name.to_string_lossy();
|
||||||
|
|
||||||
|
// The directory name should only contain the name of the package. The last two parts
|
||||||
|
// when splitting on a dash are the pkgver and pkgrel, so we trim those
|
||||||
|
let name_parts = file_name.split('-').collect::<Vec<_>>();
|
||||||
|
let name = name_parts[..name_parts.len() - 2].join("-");
|
||||||
|
|
||||||
|
if name == pkg_name {
|
||||||
|
fs::remove_dir_all(entry.path())?;
|
||||||
|
|
||||||
|
// Also remove the old package archive
|
||||||
|
let arch_repo_pkg_dir = self.pkg_dir.join(repo).join(arch);
|
||||||
|
|
||||||
|
for entry in arch_repo_pkg_dir.read_dir()? {
|
||||||
|
let entry = entry?;
|
||||||
|
let file_name = entry.file_name();
|
||||||
|
let file_name = file_name.to_string_lossy();
|
||||||
|
|
||||||
|
// Same trick, but for package files, we also need to trim the arch
|
||||||
|
let name_parts = file_name.split('-').collect::<Vec<_>>();
|
||||||
|
let name = name_parts[..name_parts.len() - 3].join("-");
|
||||||
|
|
||||||
|
if name == pkg_name {
|
||||||
|
fs::remove_file(entry.path())?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Ok(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_pkg_from_path<P: AsRef<Path>>(&mut self, repo: &str, path: P) -> io::Result<()> {
|
||||||
|
let mut pkg = Package::open(&path)?;
|
||||||
|
pkg.calculate_checksum()?;
|
||||||
|
|
||||||
|
let archs = self.add_pkg_in_repo(repo, &pkg)?;
|
||||||
|
|
||||||
|
// We add the package to each architecture it was added to by hard-linking the provided
|
||||||
|
// package file. This prevents storing a package of type "any" multiple times on disk.
|
||||||
|
for arch in archs {
|
||||||
|
let arch_repo_pkg_path = self.pkg_dir.join(repo).join(arch);
|
||||||
|
let dest_pkg_path = arch_repo_pkg_path.join(pkg.file_name());
|
||||||
|
|
||||||
|
fs::create_dir_all(&arch_repo_pkg_path)?;
|
||||||
|
fs::hard_link(&path, dest_pkg_path)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
fs::remove_file(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Add a package to the given repo, returning to what architectures the package was added.
|
||||||
|
pub fn add_pkg_in_repo(&mut self, repo: &str, pkg: &Package) -> io::Result<Vec<String>> {
|
||||||
|
if pkg.info.arch != "any" {
|
||||||
|
self.add_pkg_in_arch_repo(repo, &pkg.info.arch, pkg)?;
|
||||||
|
|
||||||
|
return Ok(vec![String::from(&pkg.info.arch)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_pkg_in_arch_repo(
|
||||||
|
&mut self,
|
||||||
|
repo: &str,
|
||||||
|
arch: &str,
|
||||||
|
pkg: &Package,
|
||||||
|
) -> io::Result<()> {
|
||||||
|
let pkg_dir = self
|
||||||
|
.repo_dir
|
||||||
|
.join(repo)
|
||||||
|
.join(arch)
|
||||||
|
.join(format!("{}-{}", pkg.info.name, pkg.info.version));
|
||||||
|
|
||||||
|
// We first remove the previous version of the package, if present
|
||||||
|
self.remove_pkg_from_arch_repo(repo, arch, &pkg.info.name)?;
|
||||||
|
|
||||||
|
fs::create_dir_all(&pkg_dir)?;
|
||||||
|
|
||||||
|
let mut desc_file = fs::File::create(pkg_dir.join("desc"))?;
|
||||||
|
pkg.write_desc(&mut desc_file)?;
|
||||||
|
|
||||||
|
let mut files_file = fs::File::create(pkg_dir.join("files"))?;
|
||||||
|
pkg.write_files(&mut files_file)?;
|
||||||
|
|
||||||
|
self.sync(repo, arch)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,7 @@ use futures::StreamExt;
|
||||||
use futures::TryFutureExt;
|
use futures::TryFutureExt;
|
||||||
use futures::TryStreamExt;
|
use futures::TryStreamExt;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
|
use std::sync::Arc;
|
||||||
use tokio::{fs, io, io::AsyncWriteExt};
|
use tokio::{fs, io, io::AsyncWriteExt};
|
||||||
use tower_http::services::ServeDir;
|
use tower_http::services::ServeDir;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
@ -53,11 +54,15 @@ async fn post_package_archive(
|
||||||
.await?;
|
.await?;
|
||||||
}
|
}
|
||||||
|
|
||||||
let pkg = tokio::task::spawn_blocking(move || package::Package::open(&path))
|
let clone = Arc::clone(&global.repo_manager);
|
||||||
.map_err(|_| StatusCode::INTERNAL_SERVER_ERROR)
|
tokio::task::spawn_blocking(move || clone.write().unwrap().add_pkg_from_path(&repo, &path))
|
||||||
.await?;
|
.map_err(|err| {
|
||||||
|
println!("{}", err);
|
||||||
println!("{:?}", pkg);
|
StatusCode::INTERNAL_SERVER_ERROR
|
||||||
|
})
|
||||||
Ok(())
|
.await?
|
||||||
|
.map_err(|err| {
|
||||||
|
println!("{}", err);
|
||||||
|
StatusCode::INTERNAL_SERVER_ERROR
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,43 +2,43 @@ use libarchive::read::{Archive, Builder};
|
||||||
use libarchive::{Entry, ReadFilter};
|
use libarchive::{Entry, ReadFilter};
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::io::{self, BufRead, BufReader, Read};
|
use std::io::{self, BufRead, BufReader, BufWriter, Read, Write};
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
const IGNORED_FILES: [&str; 5] = [".BUILDINFO", ".INSTALL", ".MTREE", ".PKGINFO", ".CHANGELOG"];
|
const IGNORED_FILES: [&str; 5] = [".BUILDINFO", ".INSTALL", ".MTREE", ".PKGINFO", ".CHANGELOG"];
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Package {
|
pub struct Package {
|
||||||
path: PathBuf,
|
pub path: PathBuf,
|
||||||
info: PkgInfo,
|
pub info: PkgInfo,
|
||||||
files: Vec<PathBuf>,
|
pub files: Vec<PathBuf>,
|
||||||
compression: ReadFilter,
|
pub compression: ReadFilter,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug, Default)]
|
||||||
pub struct PkgInfo {
|
pub struct PkgInfo {
|
||||||
name: String,
|
pub name: String,
|
||||||
base: String,
|
pub base: String,
|
||||||
version: String,
|
pub version: String,
|
||||||
description: String,
|
pub description: String,
|
||||||
size: u64,
|
pub size: u64,
|
||||||
csize: u64,
|
pub csize: u64,
|
||||||
url: String,
|
pub url: String,
|
||||||
arch: String,
|
pub arch: String,
|
||||||
build_date: i64,
|
pub build_date: i64,
|
||||||
packager: String,
|
pub packager: String,
|
||||||
pgpsig: String,
|
pub pgpsig: String,
|
||||||
pgpsigsize: i64,
|
pub pgpsigsize: i64,
|
||||||
groups: Vec<String>,
|
pub groups: Vec<String>,
|
||||||
licenses: Vec<String>,
|
pub licenses: Vec<String>,
|
||||||
replaces: Vec<String>,
|
pub replaces: Vec<String>,
|
||||||
depends: Vec<String>,
|
pub depends: Vec<String>,
|
||||||
conflicts: Vec<String>,
|
pub conflicts: Vec<String>,
|
||||||
provides: Vec<String>,
|
pub provides: Vec<String>,
|
||||||
optdepends: Vec<String>,
|
pub optdepends: Vec<String>,
|
||||||
makedepends: Vec<String>,
|
pub makedepends: Vec<String>,
|
||||||
checkdepends: Vec<String>,
|
pub checkdepends: Vec<String>,
|
||||||
sha256sum: Option<String>,
|
pub sha256sum: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq)]
|
#[derive(Debug, PartialEq, Eq)]
|
||||||
|
@ -193,4 +193,65 @@ impl Package {
|
||||||
self.compression.extension().unwrap()
|
self.compression.extension().unwrap()
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Write the formatted desc file to the provided writer
|
||||||
|
pub fn write_desc<W: Write>(&self, w: &mut W) -> io::Result<()> {
|
||||||
|
// We write a lot of small strings to the writer, so wrapping it in a BufWriter is
|
||||||
|
// beneficial
|
||||||
|
let mut w = BufWriter::new(w);
|
||||||
|
|
||||||
|
let info = &self.info;
|
||||||
|
|
||||||
|
writeln!(w, "%FILENAME%\n{}", self.file_name())?;
|
||||||
|
|
||||||
|
let mut write = |key: &str, value: &str| {
|
||||||
|
if !value.is_empty() {
|
||||||
|
writeln!(w, "\n%{}%\n{}", key, value)
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
write("NAME", &info.name)?;
|
||||||
|
write("BASE", &info.base)?;
|
||||||
|
write("VERSION", &info.version)?;
|
||||||
|
write("DESC", &info.description)?;
|
||||||
|
write("GROUPS", &info.groups.join("\n"))?;
|
||||||
|
write("CSIZE", &info.csize.to_string())?;
|
||||||
|
write("ISIZE", &info.size.to_string())?;
|
||||||
|
|
||||||
|
if let Some(checksum) = &info.sha256sum {
|
||||||
|
write("SHA256SUM", checksum)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
write("URL", &info.url)?;
|
||||||
|
write("LICENSE", &info.licenses.join("\n"))?;
|
||||||
|
write("ARCH", &info.arch)?;
|
||||||
|
write("BUILDDATE", &info.build_date.to_string())?;
|
||||||
|
write("PACKAGER", &info.packager)?;
|
||||||
|
|
||||||
|
write("REPLACES", &info.replaces.join("\n"))?;
|
||||||
|
write("CONFLICTS", &info.conflicts.join("\n"))?;
|
||||||
|
write("PROVIDES", &info.provides.join("\n"))?;
|
||||||
|
write("DEPENDS", &info.depends.join("\n"))?;
|
||||||
|
write("OPTDEPENDS", &info.optdepends.join("\n"))?;
|
||||||
|
write("MAKEDEPENDS", &info.makedepends.join("\n"))?;
|
||||||
|
write("CHECKDEPENDS", &info.checkdepends.join("\n"))?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn write_files<W: Write>(&self, w: &mut W) -> io::Result<()> {
|
||||||
|
// We write a lot of small strings to the writer, so wrapping it in a BufWriter is
|
||||||
|
// beneficial
|
||||||
|
let mut w = BufWriter::new(w);
|
||||||
|
|
||||||
|
writeln!(w, "%FILES%")?;
|
||||||
|
|
||||||
|
for file in &self.files {
|
||||||
|
writeln!(w, "{}", file.to_string_lossy())?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue