asyncgit 0.28.1

allow using git2 in a asynchronous context
Documentation
use crate::error::Result;
use git2::Repository;
use scopetime::scope_time;
use serde::{Deserialize, Serialize};

use super::{repository::repo, RepoPath};

// see https://git-scm.com/docs/git-config#Documentation/git-config.txt-statusshowUntrackedFiles
/// represents the `status.showUntrackedFiles` git config state
#[derive(
	Hash, Copy, Clone, Default, PartialEq, Eq, Serialize, Deserialize,
)]
pub enum ShowUntrackedFilesConfig {
	///
	#[default]
	No,
	///
	Normal,
	///
	All,
}

impl ShowUntrackedFilesConfig {
	///
	pub const fn include_none(self) -> bool {
		matches!(self, Self::No)
	}

	///
	pub const fn include_untracked(self) -> bool {
		matches!(self, Self::Normal | Self::All)
	}

	///
	pub const fn recurse_untracked_dirs(self) -> bool {
		matches!(self, Self::All)
	}
}

pub fn untracked_files_config_repo(
	repo: &Repository,
) -> Result<ShowUntrackedFilesConfig> {
	let show_untracked_files =
		get_config_string_repo(repo, "status.showUntrackedFiles")?;

	if let Some(show_untracked_files) = show_untracked_files {
		if &show_untracked_files == "no" {
			return Ok(ShowUntrackedFilesConfig::No);
		} else if &show_untracked_files == "normal" {
			return Ok(ShowUntrackedFilesConfig::Normal);
		}
	}

	// This does not reflect how git works according to its docs that say: "If this variable is not
	// specified, it defaults to `normal`."
	//
	// https://git-scm.com/docs/git-config#Documentation/git-config.txt-statusshowUntrackedFiles
	//
	// Note that this might become less relevant over time as more code gets migrated to `gitoxide`
	// because `gitoxide` respects `status.showUntrackedFiles` by default.
	Ok(ShowUntrackedFilesConfig::All)
}

// see https://git-scm.com/docs/git-config#Documentation/git-config.txt-pushdefault
/// represents `push.default` git config
#[derive(PartialEq, Default, Eq)]
pub enum PushDefaultStrategyConfig {
	Nothing,
	Current,
	Upstream,
	#[default]
	Simple,
	Matching,
}

impl<'a> TryFrom<&'a str> for PushDefaultStrategyConfig {
	type Error = crate::Error;
	fn try_from(
		value: &'a str,
	) -> std::result::Result<Self, Self::Error> {
		match value {
			"nothing" => Ok(Self::Nothing),
			"current" => Ok(Self::Current),
			"upstream" | "tracking" => Ok(Self::Upstream),
			"simple" => Ok(Self::Simple),
			"matching" => Ok(Self::Matching),
			_ => Err(crate::Error::GitConfig(format!(
				"malformed value for push.default: {value}, must be one of nothing, matching, simple, upstream or current"
			))),
		}
	}
}

pub fn push_default_strategy_config_repo(
	repo: &Repository,
) -> Result<PushDefaultStrategyConfig> {
	(get_config_string_repo(repo, "push.default")?).map_or_else(
		|| Ok(PushDefaultStrategyConfig::default()),
		|entry_str| {
			PushDefaultStrategyConfig::try_from(entry_str.as_str())
		},
	)
}

///
pub fn untracked_files_config(
	repo_path: &RepoPath,
) -> Result<ShowUntrackedFilesConfig> {
	let repo = repo(repo_path)?;
	untracked_files_config_repo(&repo)
}

/// get string from config
pub fn get_config_string(
	repo_path: &RepoPath,
	key: &str,
) -> Result<Option<String>> {
	let repo = repo(repo_path)?;
	get_config_string_repo(&repo, key)
}

pub fn get_config_string_repo(
	repo: &Repository,
	key: &str,
) -> Result<Option<String>> {
	scope_time!("get_config_string_repo");

	let cfg = repo.config()?;

	// this code doesn't match what the doc says regarding what
	// gets returned when but it actually works
	let entry_res = cfg.get_entry(key);

	let Ok(entry) = entry_res else {
		return Ok(None);
	};

	if entry.has_value() {
		Ok(entry.value().map(std::string::ToString::to_string))
	} else {
		Ok(None)
	}
}

#[cfg(test)]
mod tests {
	use super::*;
	use crate::sync::tests::repo_init;

	#[test]
	fn test_get_config() {
		let bad_dir_cfg = get_config_string(
			&"oodly_noodly".into(),
			"this.doesnt.exist",
		);
		assert!(bad_dir_cfg.is_err());

		let (_td, repo) = repo_init().unwrap();
		let path = repo.path();
		let rpath = path.as_os_str().to_str().unwrap();
		let bad_cfg =
			get_config_string(&rpath.into(), "this.doesnt.exist");
		assert!(bad_cfg.is_ok());
		assert!(bad_cfg.unwrap().is_none());
		// repo init sets user.name
		let good_cfg = get_config_string(&rpath.into(), "user.name");
		assert!(good_cfg.is_ok());
		assert!(good_cfg.unwrap().is_some());
	}
}