oxen-cli 0.50.7

Oxen is a fast, unstructured data version control, to help version large machine learning datasets written in Rust.
use anyhow::Context;
use async_trait::async_trait;
use clap::{Arg, ArgMatches, Command};
use colored::Colorize;
use minus::Pager;
use std::fmt::Write;
use time::format_description;

use liboxen::model::LocalRepository;
use liboxen::repositories;

use crate::cmd::RunCmd;

pub struct LogCmd;

const NAME: &str = "log";

fn write_to_pager(output: &mut Pager, text: &str) -> Result<(), anyhow::Error> {
    writeln!(output, "{text}").context("Could not write to pager.")
}

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

    fn args(&self) -> Command {
        Command::new(NAME)
            .about("See log of commits")
            .arg(
                Arg::new("revision")
                    .long("revision")
                    .help("The commit or branch id you want to get history from. Defaults to main.")
                    .action(clap::ArgAction::Set),
            )
            .arg(
                Arg::new("number")
                    .long("number")
                    .short('n')
                    .help("Number of commits to show")
                    .default_value("20"),
            )
    }

    async fn run(&self, args: &ArgMatches) -> Result<(), anyhow::Error> {
        // Look up from the current dir for .oxen directory
        let repo = LocalRepository::from_current_dir()?;

        let num_commits = args
            .get_one::<String>("number")
            .expect("Must supply number")
            .parse::<usize>()
            .expect("number must be a valid integer.");
        let revision = args.get_one::<String>("revision").map(String::from);
        self.log_commits(&repo, revision, num_commits).await?;

        Ok(())
    }
}

impl LogCmd {
    pub async fn log_commits(
        &self,
        repo: &LocalRepository,
        revision: Option<String>,
        num_commits: usize,
    ) -> Result<(), anyhow::Error> {
        let revision = match revision {
            Some(revision) => revision,
            None => repositories::commits::head_commit(repo)?.id,
        };
        let commits = repositories::commits::list_from(repo, &revision)?;
        let commits = commits.iter().take(num_commits);

        // Fri, 21 Oct 2022 16:08:39 -0700
        let format = format_description::parse(
            "[weekday], [day] [month repr:long] [year] [hour]:[minute]:[second] [offset_hour sign:mandatory]",
        ).unwrap();

        let mut output = Pager::new();

        for commit in commits {
            let commit_id_str = format!("commit {}", commit.id).yellow();
            write_to_pager(&mut output, &format!("{commit_id_str}\n"))?;
            write_to_pager(&mut output, &format!("Author: {}", commit.author))?;
            write_to_pager(
                &mut output,
                &format!("Date:   {}\n", commit.timestamp.format(&format).unwrap()),
            )?;
            write_to_pager(&mut output, &format!("    {}\n", commit.message))?;
        }

        match minus::page_all(output) {
            Ok(_) => {}
            Err(e) => {
                eprintln!("Error while paging: {e}");
            }
        }
        Ok(())
    }
}