From 87fc7035eb32a32a718c0e9d885eca01b339c042 Mon Sep 17 00:00:00 2001 From: Paul Campbell Date: Thu, 17 Oct 2024 20:32:42 +0100 Subject: [PATCH] feat: find upstream remote when using ssh The `upstream_remote` function was relying on `url::Url::parse` to extract the `owner` and `repo` from the `url`. But that only works when the repo is cloned using a URL, e.g. `https://github.com/orhun/git-cliff.git`. However, this would fail to parse when cloned using SSH, e.g. `git@github.com:orhun/git-cliff.git`. If the url::URL::parser fails, we now try to parse an SSH remote in the format `git@hostname:owner/repo.git`. The error from `upstream_remote` also notes that a posible reason for it failing would be that the `HEAD` is detached. --- CONTRIBUTING.md | 2 + git-cliff-core/src/repo.rs | 126 +++++++++++++++++++++++++++++++------ 2 files changed, 109 insertions(+), 19 deletions(-) diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 3812f70ec8..a2024c9d55 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -15,6 +15,8 @@ Note that we have a [Code of Conduct](./CODE_OF_CONDUCT.md), please follow it in ```sh git clone https://github.com/{username}/git-cliff && cd git-cliff +# OR +git clone git@github.com:{username}/git-cliff && cd git-cliff ``` To ensure the successful execution of the tests, it is essential to fetch the tags as follows: diff --git a/git-cliff-core/src/repo.rs b/git-cliff-core/src/repo.rs index 170ab1b413..bedc36e88f 100644 --- a/git-cliff-core/src/repo.rs +++ b/git-cliff-core/src/repo.rs @@ -399,6 +399,8 @@ impl Repository { /// /// Find the branch that HEAD points to, and read the remote configured for /// that branch returns the remote and the name of the local branch. + /// + /// Note: HEAD must not be detached. pub fn upstream_remote(&self) -> Result { for branch in self.inner.branches(Some(BranchType::Local))? { let branch = branch?.0; @@ -424,30 +426,98 @@ impl Repository { })? .to_string(); trace!("Upstream URL: {url}"); - let url = Url::parse(&url)?; - let segments: Vec<&str> = url - .path_segments() - .ok_or_else(|| { - Error::RepoError(String::from("failed to get URL segments")) - })? - .rev() - .collect(); - if let (Some(owner), Some(repo)) = - (segments.get(1), segments.first()) - { - return Ok(Remote { - owner: owner.to_string(), - repo: repo.trim_end_matches(".git").to_string(), - token: None, - is_custom: false, - }); - } + return find_remote(&url); } } - Err(Error::RepoError(String::from("no remotes configured"))) + Err(Error::RepoError(String::from( + "no remotes configured or HEAD is detached", + ))) } } +fn find_remote(url: &str) -> Result { + url_path_segments(url).or_else(|err| { + if url.contains("@") && url.contains(":") && url.contains("/") { + ssh_path_segments(url) + } else { + Err(err) + } + }) +} + +/// Returns the Remote from parsing the HTTPS format URL. +/// +/// This function expects the URL to be in the following format: +/// +/// https://hostname/query/path.git +/// +/// The key part is the query path, where only the last two path segments are +/// used as the owner and repo in that order. +/// +/// The returned `Remote` will contain `owner` and `repo` taken from the query +/// path. +/// +/// This function will return an `Error::UrlParseError` if the URL is malformed. +/// +/// This function will return an `Error::RepoError` if the query path has less +/// than two path segments. +fn url_path_segments(url: &str) -> Result { + let parsed_url = Url::parse(url.strip_suffix(".git").unwrap_or(url))?; + let segments: Vec<&str> = parsed_url + .path_segments() + .ok_or_else(|| Error::RepoError(String::from("failed to get URL segments")))? + .rev() + .collect(); + let [repo, owner, ..] = &segments[..] else { + return Err(Error::RepoError(String::from( + "failed to get the owner and repo", + ))); + }; + Ok(Remote { + owner: owner.to_string(), + repo: repo.to_string(), + token: None, + is_custom: false, + }) +} + +/// Returns the Remote from parsing the SSH format URL. +/// +/// This function expects the URL to be in the following format: +/// +/// git@hostname:owner/repo.git +/// +/// The key parts are the colon (:) and the path separater (/). +/// +/// The returned `Remote` will contain the `owner` and `repo` parts from the +/// URL. +/// +/// This function will return an `Error::RepoError` if the colon or separator +/// are not found. +fn ssh_path_segments(url: &str) -> Result { + let [_, owner_repo, ..] = url + .strip_suffix(".git") + .unwrap_or(url) + .split(":") + .collect::>()[..] + else { + return Err(Error::RepoError(String::from( + "failed to get the owner and repo from ssh remote (:)", + ))); + }; + let [owner, repo] = owner_repo.split("/").collect::>()[..] else { + return Err(Error::RepoError(String::from( + "failed to get the owner and repo from ssh remote (/)", + ))); + }; + Ok(Remote { + owner: owner.to_string(), + repo: repo.to_string(), + token: None, + is_custom: false, + }) +} + #[cfg(test)] mod test { use super::*; @@ -502,6 +572,24 @@ mod test { ) } + #[test] + fn http_url_repo_owner() -> Result<()> { + let url = "https://hostname.com/bob/magic.git"; + let remote = find_remote(url)?; + assert_eq!(remote.owner, "bob", "match owner"); + assert_eq!(remote.repo, "magic", "match repo"); + Ok(()) + } + + #[test] + fn ssh_url_repo_owner() -> Result<()> { + let url = "git@hostname.com:bob/magic.git"; + let remote = find_remote(url)?; + assert_eq!(remote.owner, "bob", "match owner"); + assert_eq!(remote.repo, "magic", "match repo"); + Ok(()) + } + #[test] fn get_latest_commit() -> Result<()> { let repository = get_repository()?;