aboutsummaryrefslogblamecommitdiff
path: root/src/main.rs
blob: 21d5a620c5e3c6cba9ec46ae16056d5c3ef374c1 (plain) (tree)
1
2
3
4
5
6
7
8
9

                            

                  
                      
                    
 
                 
                                  
 





                           









                                                     
                 

 


                                                             
 







                                         


                               
                                                               


                                                               
                                                                                        








                                                                                                 
 

                                         
 
            







                                                                    

 

                                                                              









                                                         
                     




                                                               




                                                                                         















                                                                                    
                      








                                             






                                          
           

                          
                             
 
                                                          
                                                
                                                        
 





                                     
 
#[macro_use]
extern crate dotenv_codegen;
extern crate git2;

use std::error::Error;
use std::path::Path;

use clap::Parser;
use git2::{Cred, RemoteCallbacks};

#[derive(Debug)]
struct Project {
    id: u64,
    default_branch: String,
}

/// Create a mirror of a repository
#[derive(Parser, Debug)]
#[command(author, version, about, long_about = None)]
struct Args {
    /// The path to save the repository to
    #[arg(short, long)]
    path: String,

    /// The URL of the repository to clone
    #[arg(short, long)]
    repo: String,
}

fn get_project_url(repo: &str) -> String {
    format!("https://{}/{}.git", dotenv!("GITLAB_URL"), repo)
}

fn get_remote_url(repo: &str) -> String {
    format!(
        "https://{}:{}@{}/{}",
        dotenv!("REMOTE_USERNAME"),
        dotenv!("REMOTE_PASSWORD"),
        dotenv!("REMOTE_URL"),
        repo
    )
}

/* Clone project from Gitlab */
fn clone_remote_project(repo: &str, branch: &str, path: &str) {
    // Callbacks for authentication
    let mut cb = RemoteCallbacks::new();
    cb.credentials(|_url, _username_from_url, _allowed_types| {
        Cred::userpass_plaintext(dotenv!("GITLAB_USERNAME"), dotenv!("GITLAB_PASSWORD"))
    });

    // Prepare fetch options
    let mut fetch_options = git2::FetchOptions::new();
    fetch_options.remote_callbacks(cb);

    // Prepare builder
    let mut builder = git2::build::RepoBuilder::new();
    builder.remote_create(|repo, name, url| repo.remote_with_fetch(name, url, "+refs/*:refs/*"));

    builder.fetch_options(fetch_options);
    builder.bare(true);

    // Clone
    let repository = builder.clone(repo, Path::new(&path)).unwrap();

    // Set default branch and config
    repository
        .set_head(&format!("refs/heads/{}", branch))
        .unwrap();
    let mut config = repository.config().unwrap();
    config.set_str("remote.origin.mirror", "true").unwrap();
}

/* Get project detail from Gitlab */
fn get_project_detail(project_name: &str) -> Result<Project, Box<dyn Error>> {
    let client = reqwest::blocking::Client::new();
    let encoded_name = urlencoding::encode(project_name);

    let api_url = format!(
        "https://{}/api/v4/projects/{}",
        dotenv!("GITLAB_URL"),
        encoded_name
    );

    let response = client
        .get(api_url)
        .bearer_auth(dotenv!("GITLAB_TOKEN"))
        .send()?;

    if response.status().is_success() {
        let project_info: serde_json::Value = response.json()?;

        Ok(Project {
            id: project_info["id"].as_u64().unwrap(),
            default_branch: project_info["default_branch"].as_str().unwrap().to_string(),
        })
    } else {
        Err("Failed to get project ID".into())
    }
}

/* Add mirror to Gitlab */
fn add_gitlab_mirror(repo_id: u64, mirror_url: &str) -> Result<(), Box<dyn Error>> {
    let client = reqwest::blocking::Client::new();

    let api_url = format!(
        "https://{}/api/v4/projects/{}/remote_mirrors",
        dotenv!("GITLAB_URL"),
        repo_id
    );

    let response = client
        .post(api_url)
        .bearer_auth(dotenv!("GITLAB_TOKEN"))
        .json(&serde_json::json!({
            "url": mirror_url,
            "enabled": true,
            "keep_divergent_refs": false,
            "only_protected_branches": false,
        }))
        .send()?;

    if response.status().is_success() {
        Ok(())
    } else {
        Err("Failed to add mirror".into())
    }
}

fn main() {
    dotenv::dotenv().ok();

    let args = Args::parse();

    let project = get_project_detail(&args.repo).unwrap();
    let mirror_url = get_remote_url(&args.path);
    add_gitlab_mirror(project.id, &mirror_url).unwrap();

    clone_remote_project(
        &get_project_url(&args.repo),
        &project.default_branch,
        &args.path,
    );
    println!("Done!")
}