oxen-cli 0.50.4

Oxen is a fast, unstructured data version control, to help version large machine learning datasets written in Rust.
use std::path::PathBuf;

use async_trait::async_trait;
use clap::{Arg, Command};
use liboxen::error::OxenError;

use crate::util;
use liboxen::model::LocalRepository;
use liboxen::opts::AddOpts;
use liboxen::repositories;

use crate::cmd::RunCmd;
use crate::helpers::check_repo_migration_needed;

pub const ADD: &str = "add";

pub struct AddCmd;

pub fn add_args() -> Command {
    // Setups the CLI args for the add command
    Command::new(ADD)
        .about("Adds the specified files or directories")
        .arg(
            Arg::new("files")
                .required(true)
                .action(clap::ArgAction::Append),
        )
}

#[async_trait]
impl RunCmd for AddCmd {
    fn name(&self) -> &str {
        ADD
    }

    fn args(&self) -> Command {
        // Setups the CLI args for the command
        add_args()
    }

    async fn run(&self, args: &clap::ArgMatches) -> Result<(), anyhow::Error> {
        // Parse Args
        let paths: Vec<PathBuf> = args
            .get_many::<String>("files")
            .expect("Must supply files")
            .map(|p| -> Result<PathBuf, OxenError> {
                let current_dir = std::env::current_dir()?;
                let joined_path = current_dir.join(p);

                util::fs::canonicalize(&joined_path).or_else(|_| Ok(joined_path))
            })
            .collect::<Result<Vec<PathBuf>, OxenError>>()?;

        let opts = AddOpts {
            paths,
            is_remote: false,
            directory: None,
        };

        // Recursively look up from the current dir for .oxen directory
        let repo = LocalRepository::from_current_dir()?;
        check_repo_migration_needed(&repo)?;

        for path in &opts.paths {
            repositories::add(&repo, path).await?;
        }

        Ok(())
    }
}