init
This commit is contained in:
		
							
								
								
									
										124
									
								
								src/api/backend.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								src/api/backend.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,124 @@
 | 
			
		||||
use crate::aur::aur::get_info_by_name;
 | 
			
		||||
use crate::builder::types::Action;
 | 
			
		||||
use crate::db::packages;
 | 
			
		||||
use crate::query_aur;
 | 
			
		||||
use rocket::serde::json::Json;
 | 
			
		||||
use rocket::serde::{Deserialize, Serialize};
 | 
			
		||||
use rocket::State;
 | 
			
		||||
use rocket::{get, post, Route};
 | 
			
		||||
use rocket_okapi::okapi::schemars;
 | 
			
		||||
use rocket_okapi::{openapi, openapi_get_routes, JsonSchema};
 | 
			
		||||
use sea_orm::{ActiveModelTrait, DatabaseConnection, Set};
 | 
			
		||||
use sea_orm::{DeleteResult, EntityTrait, ModelTrait};
 | 
			
		||||
use tokio::sync::broadcast::Sender;
 | 
			
		||||
 | 
			
		||||
use crate::db::prelude::Packages;
 | 
			
		||||
use crate::repo::repo::remove_pkg;
 | 
			
		||||
 | 
			
		||||
#[derive(Serialize, JsonSchema)]
 | 
			
		||||
#[serde(crate = "rocket::serde")]
 | 
			
		||||
struct ApiPackage {
 | 
			
		||||
    name: String,
 | 
			
		||||
    version: String,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[openapi(tag = "test")]
 | 
			
		||||
#[get("/search?<query>")]
 | 
			
		||||
async fn search(query: &str) -> Result<Json<Vec<ApiPackage>>, String> {
 | 
			
		||||
    match query_aur(query).await {
 | 
			
		||||
        Ok(v) => {
 | 
			
		||||
            let mapped = v
 | 
			
		||||
                .iter()
 | 
			
		||||
                .map(|x| ApiPackage {
 | 
			
		||||
                    name: x.name.clone(),
 | 
			
		||||
                    version: x.version.clone(),
 | 
			
		||||
                })
 | 
			
		||||
                .collect();
 | 
			
		||||
            return Ok(Json(mapped));
 | 
			
		||||
        }
 | 
			
		||||
        Err(e) => {
 | 
			
		||||
            return Err(format!("{}", e));
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[openapi(tag = "test")]
 | 
			
		||||
#[get("/packages/list")]
 | 
			
		||||
async fn package_list(
 | 
			
		||||
    db: &State<DatabaseConnection>,
 | 
			
		||||
) -> Result<Json<Vec<packages::Model>>, String> {
 | 
			
		||||
    let db = db as &DatabaseConnection;
 | 
			
		||||
 | 
			
		||||
    let all: Vec<packages::Model> = Packages::find().all(db).await.unwrap();
 | 
			
		||||
 | 
			
		||||
    Ok(Json(all))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Deserialize, JsonSchema)]
 | 
			
		||||
#[serde(crate = "rocket::serde")]
 | 
			
		||||
struct AddBody {
 | 
			
		||||
    name: String,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[openapi(tag = "test")]
 | 
			
		||||
#[post("/packages/add", data = "<input>")]
 | 
			
		||||
async fn package_add(
 | 
			
		||||
    db: &State<DatabaseConnection>,
 | 
			
		||||
    input: Json<AddBody>,
 | 
			
		||||
    tx: &State<Sender<Action>>,
 | 
			
		||||
) -> Result<(), String> {
 | 
			
		||||
    let db = db as &DatabaseConnection;
 | 
			
		||||
 | 
			
		||||
    let pkg_name = &input.name;
 | 
			
		||||
 | 
			
		||||
    let pkg = get_info_by_name(pkg_name)
 | 
			
		||||
        .await
 | 
			
		||||
        .map_err(|_| "couldn't download package metadata".to_string())?;
 | 
			
		||||
 | 
			
		||||
    let new_package = packages::ActiveModel {
 | 
			
		||||
        name: Set(pkg_name.clone()),
 | 
			
		||||
        version: Set(pkg.version.clone()),
 | 
			
		||||
        ..Default::default()
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    let t = new_package.save(db).await.expect("TODO: panic message");
 | 
			
		||||
 | 
			
		||||
    let _ = tx.send(Action::Build(
 | 
			
		||||
        pkg.name,
 | 
			
		||||
        pkg.version,
 | 
			
		||||
        pkg.url_path.unwrap(),
 | 
			
		||||
        t.id.unwrap(),
 | 
			
		||||
    ));
 | 
			
		||||
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Deserialize, JsonSchema)]
 | 
			
		||||
#[serde(crate = "rocket::serde")]
 | 
			
		||||
struct DelBody {
 | 
			
		||||
    id: i32,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[openapi(tag = "test")]
 | 
			
		||||
#[post("/packages/delete", data = "<input>")]
 | 
			
		||||
async fn package_del(db: &State<DatabaseConnection>, input: Json<DelBody>) -> Result<(), String> {
 | 
			
		||||
    let db = db as &DatabaseConnection;
 | 
			
		||||
    let pkg_id = &input.id;
 | 
			
		||||
 | 
			
		||||
    let pkg = Packages::find_by_id(*pkg_id)
 | 
			
		||||
        .one(db)
 | 
			
		||||
        .await
 | 
			
		||||
        .unwrap()
 | 
			
		||||
        .unwrap();
 | 
			
		||||
 | 
			
		||||
    // remove folders
 | 
			
		||||
    remove_pkg(pkg.name.to_string(), pkg.version.to_string()).await;
 | 
			
		||||
 | 
			
		||||
    // remove package db entry
 | 
			
		||||
    let res: DeleteResult = pkg.delete(db).await.unwrap();
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
pub fn build_api() -> Vec<Route> {
 | 
			
		||||
    openapi_get_routes![search, package_list, package_add, package_del]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										2
									
								
								src/api/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/api/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
			
		||||
pub mod backend;
 | 
			
		||||
pub mod repository;
 | 
			
		||||
							
								
								
									
										5
									
								
								src/api/repository.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/api/repository.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
use rocket::fs::FileServer;
 | 
			
		||||
 | 
			
		||||
pub fn build_api() -> FileServer {
 | 
			
		||||
    FileServer::from("./repo")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										99
									
								
								src/aur/aur.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								src/aur/aur.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,99 @@
 | 
			
		||||
use anyhow::anyhow;
 | 
			
		||||
use aur_rs::{Package, Request};
 | 
			
		||||
use flate2::bufread::GzDecoder;
 | 
			
		||||
use std::fs;
 | 
			
		||||
use std::path::Path;
 | 
			
		||||
use tar::Archive;
 | 
			
		||||
 | 
			
		||||
pub async fn query_aur(query: &str) -> anyhow::Result<Vec<Package>> {
 | 
			
		||||
    let request = Request::default();
 | 
			
		||||
    let response = request.search_package_by_name(query).await;
 | 
			
		||||
 | 
			
		||||
    let response = match response {
 | 
			
		||||
        Ok(v) => v,
 | 
			
		||||
        Err(_) => {
 | 
			
		||||
            return Err(anyhow!("failed to search"));
 | 
			
		||||
        }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    let mut response = response.results;
 | 
			
		||||
    response.sort_by(|x, x1| x.popularity.partial_cmp(&x1.popularity).unwrap().reverse());
 | 
			
		||||
 | 
			
		||||
    Ok(response)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
pub async fn get_info_by_name(pkg_name: &str) -> anyhow::Result<Package> {
 | 
			
		||||
    let request = Request::default();
 | 
			
		||||
    let response = request.search_info_by_name(pkg_name).await;
 | 
			
		||||
 | 
			
		||||
    let mut response = match response {
 | 
			
		||||
        Ok(v) => v,
 | 
			
		||||
        Err(_) => {
 | 
			
		||||
            return Err(anyhow!("failed to get package"));
 | 
			
		||||
        }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    let response = match response.results.pop() {
 | 
			
		||||
        None => {
 | 
			
		||||
            return Err(anyhow!("no package found"));
 | 
			
		||||
        }
 | 
			
		||||
        Some(v) => v,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    Ok(response)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
pub async fn download_pkgbuild(url: &str, dest_dir: &str) -> anyhow::Result<String> {
 | 
			
		||||
    let (file_data, file_name) = match download_file(url).await {
 | 
			
		||||
        Ok(data) => data,
 | 
			
		||||
        Err(e) => {
 | 
			
		||||
            return Err(anyhow!("Error downloading file: {}", e));
 | 
			
		||||
        }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    // Check if the directory exists
 | 
			
		||||
    if !fs::metadata(dest_dir).is_ok() {
 | 
			
		||||
        // Create the directory if it does not exist
 | 
			
		||||
        fs::create_dir(dest_dir)?;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    unpack_tar_gz(&file_data, dest_dir)?;
 | 
			
		||||
    Ok(file_name)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async fn download_file(url: &str) -> anyhow::Result<(Vec<u8>, String)> {
 | 
			
		||||
    let response = reqwest::get(url).await?;
 | 
			
		||||
 | 
			
		||||
    // extract name of file without extension
 | 
			
		||||
    // todo might be also possible here to use package name
 | 
			
		||||
    let t = response
 | 
			
		||||
        .url()
 | 
			
		||||
        .path_segments()
 | 
			
		||||
        .and_then(|segments| segments.last())
 | 
			
		||||
        .ok_or(anyhow!("no segments"))?
 | 
			
		||||
        .split(".")
 | 
			
		||||
        .collect::<Vec<&str>>()
 | 
			
		||||
        .first()
 | 
			
		||||
        .ok_or(anyhow!(""))?
 | 
			
		||||
        .to_string();
 | 
			
		||||
 | 
			
		||||
    println!("{}", t);
 | 
			
		||||
 | 
			
		||||
    let r = response.bytes().await?;
 | 
			
		||||
    Ok((r.to_vec(), t))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fn unpack_tar_gz(data: &[u8], target_dir: &str) -> anyhow::Result<()> {
 | 
			
		||||
    let tar = GzDecoder::new(data);
 | 
			
		||||
    let mut archive = Archive::new(tar);
 | 
			
		||||
 | 
			
		||||
    for entry in archive.entries()? {
 | 
			
		||||
        let mut entry = entry?;
 | 
			
		||||
        let path = entry.path()?;
 | 
			
		||||
        let entry_path = Path::new(target_dir).join(path);
 | 
			
		||||
 | 
			
		||||
        entry.unpack(entry_path)?;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								src/aur/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/aur/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
pub mod aur;
 | 
			
		||||
							
								
								
									
										39
									
								
								src/builder/builder.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/builder/builder.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
use crate::builder::types::Action;
 | 
			
		||||
use crate::db::packages;
 | 
			
		||||
use crate::db::prelude::Packages;
 | 
			
		||||
use crate::repo::repo::add_pkg;
 | 
			
		||||
use sea_orm::{ActiveModelTrait, DatabaseConnection, EntityTrait, Set};
 | 
			
		||||
use tokio::sync::broadcast::Sender;
 | 
			
		||||
 | 
			
		||||
pub async fn init(db: DatabaseConnection, tx: Sender<Action>) {
 | 
			
		||||
    loop {
 | 
			
		||||
        if let Ok(_result) = tx.subscribe().recv().await {
 | 
			
		||||
            match _result {
 | 
			
		||||
                // add a package to parallel build
 | 
			
		||||
                Action::Build(name, version, url, id) => {
 | 
			
		||||
                    let db = db.clone();
 | 
			
		||||
                    tokio::spawn(async move {
 | 
			
		||||
                        match add_pkg(url, version, name).await {
 | 
			
		||||
                            Ok(_) => {
 | 
			
		||||
                                println!("successfully built package");
 | 
			
		||||
 | 
			
		||||
                                let mut pkg: packages::ActiveModel = Packages::find_by_id(id)
 | 
			
		||||
                                    .one(&db)
 | 
			
		||||
                                    .await
 | 
			
		||||
                                    .unwrap()
 | 
			
		||||
                                    .unwrap()
 | 
			
		||||
                                    .into();
 | 
			
		||||
 | 
			
		||||
                                pkg.status = Set(2);
 | 
			
		||||
                                let pkg: packages::Model = pkg.update(&db).await.unwrap();
 | 
			
		||||
                            }
 | 
			
		||||
                            Err(e) => {
 | 
			
		||||
                                println!("Error: {e}")
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    });
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										2
									
								
								src/builder/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/builder/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
			
		||||
pub mod builder;
 | 
			
		||||
pub mod types;
 | 
			
		||||
							
								
								
									
										4
									
								
								src/builder/types.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/builder/types.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
#[derive(Clone)]
 | 
			
		||||
pub enum Action {
 | 
			
		||||
    Build(String, String, String, i32),
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										49
									
								
								src/db/migration/m20220101_000001_create_table.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								src/db/migration/m20220101_000001_create_table.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,49 @@
 | 
			
		||||
use sea_orm_migration::prelude::*;
 | 
			
		||||
 | 
			
		||||
#[derive(DeriveMigrationName)]
 | 
			
		||||
pub struct Migration;
 | 
			
		||||
 | 
			
		||||
#[async_trait::async_trait]
 | 
			
		||||
impl MigrationTrait for Migration {
 | 
			
		||||
    async fn up(&self, manager: &SchemaManager) -> Result<(), DbErr> {
 | 
			
		||||
        manager
 | 
			
		||||
            .create_table(
 | 
			
		||||
                Table::create()
 | 
			
		||||
                    .table(Packages::Table)
 | 
			
		||||
                    .if_not_exists()
 | 
			
		||||
                    .col(
 | 
			
		||||
                        ColumnDef::new(Packages::Id)
 | 
			
		||||
                            .integer()
 | 
			
		||||
                            .not_null()
 | 
			
		||||
                            .auto_increment()
 | 
			
		||||
                            .primary_key(),
 | 
			
		||||
                    )
 | 
			
		||||
                    .col(ColumnDef::new(Packages::Version).string().not_null())
 | 
			
		||||
                    .col(ColumnDef::new(Packages::name).string().not_null())
 | 
			
		||||
                    .col(
 | 
			
		||||
                        ColumnDef::new(Packages::Status)
 | 
			
		||||
                            .integer()
 | 
			
		||||
                            .not_null()
 | 
			
		||||
                            .default(0),
 | 
			
		||||
                    )
 | 
			
		||||
                    .to_owned(),
 | 
			
		||||
            )
 | 
			
		||||
            .await
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async fn down(&self, manager: &SchemaManager) -> Result<(), DbErr> {
 | 
			
		||||
        manager
 | 
			
		||||
            .drop_table(Table::drop().table(Packages::Table).to_owned())
 | 
			
		||||
            .await
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Learn more at https://docs.rs/sea-query#iden
 | 
			
		||||
#[derive(Iden)]
 | 
			
		||||
enum Packages {
 | 
			
		||||
    Table,
 | 
			
		||||
    name,
 | 
			
		||||
    Version,
 | 
			
		||||
    Id,
 | 
			
		||||
    Status,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										12
									
								
								src/db/migration/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/db/migration/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
			
		||||
pub use sea_orm_migration::prelude::*;
 | 
			
		||||
 | 
			
		||||
mod m20220101_000001_create_table;
 | 
			
		||||
 | 
			
		||||
pub struct Migrator;
 | 
			
		||||
 | 
			
		||||
#[async_trait::async_trait]
 | 
			
		||||
impl MigratorTrait for Migrator {
 | 
			
		||||
    fn migrations() -> Vec<Box<dyn MigrationTrait>> {
 | 
			
		||||
        vec![Box::new(m20220101_000001_create_table::Migration)]
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										6
									
								
								src/db/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/db/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,6 @@
 | 
			
		||||
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.11.2
 | 
			
		||||
 | 
			
		||||
pub mod prelude;
 | 
			
		||||
 | 
			
		||||
pub mod migration;
 | 
			
		||||
pub mod packages;
 | 
			
		||||
							
								
								
									
										21
									
								
								src/db/packages.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/db/packages.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.11.2
 | 
			
		||||
 | 
			
		||||
use rocket::serde::Serialize;
 | 
			
		||||
use rocket_okapi::okapi::schemars;
 | 
			
		||||
use rocket_okapi::JsonSchema;
 | 
			
		||||
use sea_orm::entity::prelude::*;
 | 
			
		||||
 | 
			
		||||
#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq, Serialize, JsonSchema)]
 | 
			
		||||
#[sea_orm(table_name = "packages")]
 | 
			
		||||
pub struct Model {
 | 
			
		||||
    pub name: String,
 | 
			
		||||
    pub version: String,
 | 
			
		||||
    #[sea_orm(primary_key, auto_increment = false)]
 | 
			
		||||
    pub id: i32,
 | 
			
		||||
    pub status: i32,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
 | 
			
		||||
pub enum Relation {}
 | 
			
		||||
 | 
			
		||||
impl ActiveModelBehavior for ActiveModel {}
 | 
			
		||||
							
								
								
									
										3
									
								
								src/db/prelude.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/db/prelude.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.11.2
 | 
			
		||||
 | 
			
		||||
pub use super::packages::Entity as Packages;
 | 
			
		||||
							
								
								
									
										89
									
								
								src/main.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								src/main.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,89 @@
 | 
			
		||||
mod api;
 | 
			
		||||
mod aur;
 | 
			
		||||
mod builder;
 | 
			
		||||
mod db;
 | 
			
		||||
mod pkgbuild;
 | 
			
		||||
mod repo;
 | 
			
		||||
 | 
			
		||||
use crate::api::{backend, repository};
 | 
			
		||||
use crate::aur::aur::query_aur;
 | 
			
		||||
use crate::builder::types::Action;
 | 
			
		||||
use crate::db::migration::Migrator;
 | 
			
		||||
use rocket::config::Config;
 | 
			
		||||
use rocket::futures::future::join_all;
 | 
			
		||||
use rocket_okapi::swagger_ui::{make_swagger_ui, SwaggerUIConfig};
 | 
			
		||||
use sea_orm::{Database, DatabaseConnection};
 | 
			
		||||
use sea_orm_migration::MigratorTrait;
 | 
			
		||||
use std::fs;
 | 
			
		||||
use tokio::sync::broadcast;
 | 
			
		||||
 | 
			
		||||
fn main() {
 | 
			
		||||
    let t = tokio::runtime::Runtime::new().unwrap();
 | 
			
		||||
 | 
			
		||||
    let (tx, _) = broadcast::channel::<Action>(32);
 | 
			
		||||
 | 
			
		||||
    t.block_on(async move {
 | 
			
		||||
        //build_package("sea-orm-cli").await;
 | 
			
		||||
 | 
			
		||||
        let db: DatabaseConnection = Database::connect("sqlite://db.sqlite?mode=rwc")
 | 
			
		||||
            .await
 | 
			
		||||
            .unwrap();
 | 
			
		||||
 | 
			
		||||
        Migrator::up(&db, None).await.unwrap();
 | 
			
		||||
 | 
			
		||||
        // Check if the directory exists
 | 
			
		||||
        if !fs::metadata("./repo").is_ok() {
 | 
			
		||||
            // Create the directory if it does not exist
 | 
			
		||||
            fs::create_dir("./repo").unwrap();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        let db2 = db.clone();
 | 
			
		||||
        let tx2 = tx.clone();
 | 
			
		||||
        tokio::spawn(async move {
 | 
			
		||||
            builder::builder::init(db2, tx2).await;
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        let backend_handle = tokio::spawn(async {
 | 
			
		||||
            let mut config = Config::default();
 | 
			
		||||
            config.address = "0.0.0.0".parse().unwrap();
 | 
			
		||||
            config.port = 8081;
 | 
			
		||||
 | 
			
		||||
            let launch_result = rocket::custom(config)
 | 
			
		||||
                .manage(db)
 | 
			
		||||
                .manage(tx)
 | 
			
		||||
                .mount("/", backend::build_api())
 | 
			
		||||
                .mount(
 | 
			
		||||
                    "/docs/",
 | 
			
		||||
                    make_swagger_ui(&SwaggerUIConfig {
 | 
			
		||||
                        url: "../openapi.json".to_owned(),
 | 
			
		||||
                        ..Default::default()
 | 
			
		||||
                    }),
 | 
			
		||||
                )
 | 
			
		||||
                .launch()
 | 
			
		||||
                .await;
 | 
			
		||||
            match launch_result {
 | 
			
		||||
                Ok(_) => println!("Rocket shut down gracefully."),
 | 
			
		||||
                Err(err) => println!("Rocket had an error: {}", err),
 | 
			
		||||
            };
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        let repo_handle = tokio::spawn(async {
 | 
			
		||||
            let mut config = Config::default();
 | 
			
		||||
            config.address = "0.0.0.0".parse().unwrap();
 | 
			
		||||
            config.port = 8080;
 | 
			
		||||
 | 
			
		||||
            let launch_result = rocket::custom(config)
 | 
			
		||||
                .mount("/", repository::build_api())
 | 
			
		||||
                .launch()
 | 
			
		||||
                .await;
 | 
			
		||||
            match launch_result {
 | 
			
		||||
                Ok(_) => println!("Rocket shut down gracefully."),
 | 
			
		||||
                Err(err) => println!("Rocket had an error: {}", err),
 | 
			
		||||
            };
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        join_all([repo_handle, backend_handle]).await;
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    return;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										95
									
								
								src/pkgbuild/build.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/pkgbuild/build.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,95 @@
 | 
			
		||||
use anyhow::anyhow;
 | 
			
		||||
use std::fs;
 | 
			
		||||
use std::process::{Command, Stdio};
 | 
			
		||||
use std::time::SystemTime;
 | 
			
		||||
 | 
			
		||||
pub fn build_pkgbuild(
 | 
			
		||||
    folder_path: String,
 | 
			
		||||
    pkg_vers: &str,
 | 
			
		||||
    pkg_name: &str,
 | 
			
		||||
) -> anyhow::Result<String> {
 | 
			
		||||
    let makepkg = include_str!("../../scripts/makepkg");
 | 
			
		||||
 | 
			
		||||
    // Create a temporary file to store the bash script content
 | 
			
		||||
    let script_file = std::env::temp_dir().join("makepkg_custom.sh");
 | 
			
		||||
    fs::write(&script_file, makepkg).expect("Unable to write script to file");
 | 
			
		||||
 | 
			
		||||
    let output = Command::new("bash")
 | 
			
		||||
        .args(&[
 | 
			
		||||
            script_file.as_os_str().to_str().unwrap(),
 | 
			
		||||
            "-f",
 | 
			
		||||
            "--noconfirm",
 | 
			
		||||
            "-s",
 | 
			
		||||
            "-c",
 | 
			
		||||
        ])
 | 
			
		||||
        .current_dir(folder_path.clone())
 | 
			
		||||
        .stdout(Stdio::inherit())
 | 
			
		||||
        .spawn()
 | 
			
		||||
        .unwrap();
 | 
			
		||||
    let output = output.wait_with_output();
 | 
			
		||||
 | 
			
		||||
    match output {
 | 
			
		||||
        Ok(output) => {
 | 
			
		||||
            if output.status.success() {
 | 
			
		||||
                let stdout = String::from_utf8_lossy(&output.stdout);
 | 
			
		||||
                println!("makepkg output: {}", stdout);
 | 
			
		||||
            } else {
 | 
			
		||||
                let stderr = String::from_utf8_lossy(&output.stderr);
 | 
			
		||||
                eprintln!("makepkg error: {}", stderr);
 | 
			
		||||
 | 
			
		||||
                return Err(anyhow!("failed to build package"));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        Err(err) => {
 | 
			
		||||
            eprintln!("Failed to execute makepkg: {}", err);
 | 
			
		||||
            return Err(anyhow!("failed to build package"));
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // check if expected built dir exists
 | 
			
		||||
    let built_name = build_repo_packagename(pkg_name.to_string(), pkg_vers.to_string());
 | 
			
		||||
    if fs::metadata(format!("{folder_path}/{built_name}")).is_ok() {
 | 
			
		||||
        println!("Built {built_name}");
 | 
			
		||||
        return Ok(built_name.to_string());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // the naming might not always contain the build version
 | 
			
		||||
    // eg. mesa-git  --> match pkgname and extension if multiple return latest
 | 
			
		||||
    if let Ok(paths) = fs::read_dir(folder_path) {
 | 
			
		||||
        let mut candidate_filename: Option<String> = None;
 | 
			
		||||
        let mut candidate_timestamp = SystemTime::UNIX_EPOCH;
 | 
			
		||||
 | 
			
		||||
        for path in paths {
 | 
			
		||||
            if let Ok(path) = path {
 | 
			
		||||
                let path = path.path();
 | 
			
		||||
                if let Some(file_name) = path.file_name() {
 | 
			
		||||
                    let file_name = file_name.to_str().unwrap();
 | 
			
		||||
 | 
			
		||||
                    if file_name.ends_with("-x86_64.pkg.tar.zst") && file_name.starts_with(pkg_name)
 | 
			
		||||
                    {
 | 
			
		||||
                        if let Ok(metadata) = path.metadata() {
 | 
			
		||||
                            if let Ok(modified_time) = metadata.modified() {
 | 
			
		||||
                                // Update the candidate filename and timestamp if the current file is newer
 | 
			
		||||
                                if modified_time > candidate_timestamp {
 | 
			
		||||
                                    candidate_filename = Some(file_name.to_string());
 | 
			
		||||
                                    candidate_timestamp = modified_time;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if candidate_filename.is_some() {
 | 
			
		||||
            println!("Built {}", candidate_filename.clone().unwrap());
 | 
			
		||||
            return Ok(candidate_filename.unwrap());
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Err(anyhow!("No package built"))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
pub fn build_repo_packagename(pkg_name: String, pkg_vers: String) -> String {
 | 
			
		||||
    format!("{pkg_name}-{pkg_vers}-x86_64.pkg.tar.zst")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								src/pkgbuild/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/pkgbuild/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
pub mod build;
 | 
			
		||||
							
								
								
									
										1
									
								
								src/repo/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/repo/mod.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
pub mod repo;
 | 
			
		||||
							
								
								
									
										76
									
								
								src/repo/repo.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								src/repo/repo.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,76 @@
 | 
			
		||||
use crate::aur::aur::download_pkgbuild;
 | 
			
		||||
use crate::pkgbuild::build::{build_pkgbuild, build_repo_packagename};
 | 
			
		||||
use anyhow::anyhow;
 | 
			
		||||
use std::fs;
 | 
			
		||||
use std::process::Command;
 | 
			
		||||
 | 
			
		||||
static REPO_NAME: &str = "repo";
 | 
			
		||||
static BASEURL: &str = "https://aur.archlinux.org";
 | 
			
		||||
 | 
			
		||||
pub async fn add_pkg(url: String, version: String, name: String) -> anyhow::Result<()> {
 | 
			
		||||
    let fname = download_pkgbuild(format!("{}{}", BASEURL, url).as_str(), "./builds").await?;
 | 
			
		||||
    let pkg_file_name =
 | 
			
		||||
        build_pkgbuild(format!("./builds/{fname}"), version.as_str(), name.as_str())?;
 | 
			
		||||
 | 
			
		||||
    // todo force overwrite if file already exists
 | 
			
		||||
    fs::copy(
 | 
			
		||||
        format!("./builds/{fname}/{pkg_file_name}"),
 | 
			
		||||
        format!("./repo/{pkg_file_name}"),
 | 
			
		||||
    )?;
 | 
			
		||||
    fs::remove_file(format!("./builds/{fname}/{pkg_file_name}"))?;
 | 
			
		||||
 | 
			
		||||
    repo_add(pkg_file_name)?;
 | 
			
		||||
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fn repo_add(pkg_file_name: String) -> anyhow::Result<()> {
 | 
			
		||||
    let db_file = format!("{REPO_NAME}.db.tar.gz");
 | 
			
		||||
 | 
			
		||||
    let output = Command::new("repo-add")
 | 
			
		||||
        .args(&[db_file.clone(), pkg_file_name])
 | 
			
		||||
        .current_dir("./repo/")
 | 
			
		||||
        .output()?;
 | 
			
		||||
 | 
			
		||||
    if !output.status.success() {
 | 
			
		||||
        return Err(anyhow!(
 | 
			
		||||
            "Error exit code when repo-add: {}{}",
 | 
			
		||||
            String::from_utf8_lossy(output.stdout.as_slice()),
 | 
			
		||||
            String::from_utf8_lossy(output.stderr.as_slice())
 | 
			
		||||
        ));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    println!("{db_file} updated successfully");
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fn repo_remove(pkg_file_name: String) -> anyhow::Result<()> {
 | 
			
		||||
    let db_file = format!("{REPO_NAME}.db.tar.gz");
 | 
			
		||||
 | 
			
		||||
    let output = Command::new("repo-remove")
 | 
			
		||||
        .args(&[db_file.clone(), pkg_file_name])
 | 
			
		||||
        .current_dir("./repo/")
 | 
			
		||||
        .output()?;
 | 
			
		||||
 | 
			
		||||
    if !output.status.success() {
 | 
			
		||||
        return Err(anyhow!(
 | 
			
		||||
            "Error exit code when repo-remove: {}{}",
 | 
			
		||||
            String::from_utf8_lossy(output.stdout.as_slice()),
 | 
			
		||||
            String::from_utf8_lossy(output.stderr.as_slice())
 | 
			
		||||
        ));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    println!("{db_file} updated successfully");
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
pub async fn remove_pkg(pkg_name: String, pkg_version: String) -> anyhow::Result<()> {
 | 
			
		||||
    fs::remove_dir_all(format!("./builds/{pkg_name}"))?;
 | 
			
		||||
 | 
			
		||||
    let filename = build_repo_packagename(pkg_name.clone(), pkg_version);
 | 
			
		||||
    fs::remove_file(format!("./repo/{filename}"))?;
 | 
			
		||||
 | 
			
		||||
    repo_remove(pkg_name)?;
 | 
			
		||||
 | 
			
		||||
    Ok(())
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user