use std::str::FromStr; use std::sync::{Arc, LazyLock}; use anyhow::{bail, Context, Result}; use async_trait::async_trait; use futures::AsyncReadExt; use gpui::SharedString; use http_client::{AsyncBody, HttpClient, HttpRequestExt, Request}; use regex::Regex; use serde::Deserialize; use url::Url; use git::{ BuildCommitPermalinkParams, BuildPermalinkParams, GitHostingProvider, ParsedGitRemote, PullRequest, RemoteUrl, }; use crate::get_host_from_git_remote_url; fn pull_request_number_regex() -> &'static Regex { static PULL_REQUEST_NUMBER_REGEX: LazyLock = LazyLock::new(|| Regex::new(r"\(#(\d+)\)$").unwrap()); &PULL_REQUEST_NUMBER_REGEX } #[derive(Debug, Deserialize)] struct CommitDetails { commit: Commit, author: Option, } #[derive(Debug, Deserialize)] struct Commit { author: Author, } #[derive(Debug, Deserialize)] struct Author { email: String, } #[derive(Debug, Deserialize)] struct User { pub id: u64, pub avatar_url: String, } #[derive(Debug)] pub struct Github { name: String, base_url: Url, } impl Github { pub fn new(name: impl Into, base_url: Url) -> Self { Self { name: name.into(), base_url, } } pub fn public_instance() -> Self { Self::new("GitHub", Url::parse("https://github.com").unwrap()) } pub fn from_remote_url(remote_url: &str) -> Result { let host = get_host_from_git_remote_url(remote_url)?; if host == "github.com" { bail!("the GitHub instance is not self-hosted"); } // TODO: detecting self hosted instances by checking whether "github" is in the url or not // is not very reliable. See https://github.com/zed-industries/zed/issues/26393 for more // information. if !host.contains("github") { bail!("not a GitHub URL"); } Ok(Self::new( "GitHub Self-Hosted", Url::parse(&format!("https://{}", host))?, )) } async fn fetch_github_commit_author( &self, repo_owner: &str, repo: &str, commit: &str, client: &Arc, ) -> Result> { let Some(host) = self.base_url.host_str() else { bail!("failed to get host from github base url"); }; let url = format!("https://api.{host}/repos/{repo_owner}/{repo}/commits/{commit}"); let mut request = Request::get(&url) .header("Content-Type", "application/json") .follow_redirects(http_client::RedirectPolicy::FollowAll); if let Ok(github_token) = std::env::var("GITHUB_TOKEN") { request = request.header("Authorization", format!("Bearer {}", github_token)); } let mut response = client .send(request.body(AsyncBody::default())?) .await .with_context(|| format!("error fetching GitHub commit details at {:?}", url))?; let mut body = Vec::new(); response.body_mut().read_to_end(&mut body).await?; if response.status().is_client_error() { let text = String::from_utf8_lossy(body.as_slice()); bail!( "status error {}, response: {text:?}", response.status().as_u16() ); } let body_str = std::str::from_utf8(&body)?; serde_json::from_str::(body_str) .map(|commit| commit.author) .context("failed to deserialize GitHub commit details") } } #[async_trait] impl GitHostingProvider for Github { fn name(&self) -> String { self.name.clone() } fn base_url(&self) -> Url { self.base_url.clone() } fn supports_avatars(&self) -> bool { // Avatars are not supported for self-hosted GitHub instances // See tracking issue: https://github.com/zed-industries/zed/issues/11043 &self.name == "GitHub" } fn format_line_number(&self, line: u32) -> String { format!("L{line}") } fn format_line_numbers(&self, start_line: u32, end_line: u32) -> String { format!("L{start_line}-L{end_line}") } fn parse_remote_url(&self, url: &str) -> Option { let url = RemoteUrl::from_str(url).ok()?; let host = url.host_str()?; if host != self.base_url.host_str()? { return None; } let mut path_segments = url.path_segments()?; let owner = path_segments.next()?; let repo = path_segments.next()?.trim_end_matches(".git"); Some(ParsedGitRemote { owner: owner.into(), repo: repo.into(), }) } fn build_commit_permalink( &self, remote: &ParsedGitRemote, params: BuildCommitPermalinkParams, ) -> Url { let BuildCommitPermalinkParams { sha } = params; let ParsedGitRemote { owner, repo } = remote; self.base_url() .join(&format!("{owner}/{repo}/commit/{sha}")) .unwrap() } fn build_permalink(&self, remote: ParsedGitRemote, params: BuildPermalinkParams) -> Url { let ParsedGitRemote { owner, repo } = remote; let BuildPermalinkParams { sha, path, selection, } = params; let mut permalink = self .base_url() .join(&format!("{owner}/{repo}/blob/{sha}/{path}")) .unwrap(); if path.ends_with(".md") { permalink.set_query(Some("plain=1")); } permalink.set_fragment( selection .map(|selection| self.line_fragment(&selection)) .as_deref(), ); permalink } fn extract_pull_request(&self, remote: &ParsedGitRemote, message: &str) -> Option { let line = message.lines().next()?; let capture = pull_request_number_regex().captures(line)?; let number = capture.get(1)?.as_str().parse::().ok()?; let mut url = self.base_url(); let path = format!("/{}/{}/pull/{}", remote.owner, remote.repo, number); url.set_path(&path); Some(PullRequest { number, url }) } async fn commit_author_avatar_url( &self, repo_owner: &str, repo: &str, commit: SharedString, http_client: Arc, ) -> Result> { let commit = commit.to_string(); let avatar_url = self .fetch_github_commit_author(repo_owner, repo, &commit, &http_client) .await? .map(|author| -> Result { let mut url = Url::parse(&author.avatar_url)?; url.set_query(Some("size=128")); Ok(url) }) .transpose()?; Ok(avatar_url) } } #[cfg(test)] mod tests { use indoc::indoc; use pretty_assertions::assert_eq; use super::*; #[test] fn test_invalid_self_hosted_remote_url() { let remote_url = "git@github.com:zed-industries/zed.git"; let github = Github::from_remote_url(remote_url); assert!(github.is_err()); } #[test] fn test_from_remote_url_ssh() { let remote_url = "git@github.my-enterprise.com:zed-industries/zed.git"; let github = Github::from_remote_url(remote_url).unwrap(); assert!(!github.supports_avatars()); assert_eq!(github.name, "GitHub Self-Hosted".to_string()); assert_eq!( github.base_url, Url::parse("https://github.my-enterprise.com").unwrap() ); } #[test] fn test_from_remote_url_https() { let remote_url = "https://github.my-enterprise.com/zed-industries/zed.git"; let github = Github::from_remote_url(remote_url).unwrap(); assert!(!github.supports_avatars()); assert_eq!(github.name, "GitHub Self-Hosted".to_string()); assert_eq!( github.base_url, Url::parse("https://github.my-enterprise.com").unwrap() ); } #[test] fn test_parse_remote_url_given_self_hosted_ssh_url() { let remote_url = "git@github.my-enterprise.com:zed-industries/zed.git"; let parsed_remote = Github::from_remote_url(remote_url) .unwrap() .parse_remote_url(remote_url) .unwrap(); assert_eq!( parsed_remote, ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), } ); } #[test] fn test_parse_remote_url_given_self_hosted_https_url_with_subgroup() { let remote_url = "https://github.my-enterprise.com/zed-industries/zed.git"; let parsed_remote = Github::from_remote_url(remote_url) .unwrap() .parse_remote_url(remote_url) .unwrap(); assert_eq!( parsed_remote, ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), } ); } #[test] fn test_parse_remote_url_given_ssh_url() { let parsed_remote = Github::public_instance() .parse_remote_url("git@github.com:zed-industries/zed.git") .unwrap(); assert_eq!( parsed_remote, ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), } ); } #[test] fn test_parse_remote_url_given_https_url() { let parsed_remote = Github::public_instance() .parse_remote_url("https://github.com/zed-industries/zed.git") .unwrap(); assert_eq!( parsed_remote, ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), } ); } #[test] fn test_parse_remote_url_given_https_url_with_username() { let parsed_remote = Github::public_instance() .parse_remote_url("https://jlannister@github.com/some-org/some-repo.git") .unwrap(); assert_eq!( parsed_remote, ParsedGitRemote { owner: "some-org".into(), repo: "some-repo".into(), } ); } #[test] fn test_build_github_permalink_from_ssh_url() { let remote = ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), }; let permalink = Github::public_instance().build_permalink( remote, BuildPermalinkParams { sha: "e6ebe7974deb6bb6cc0e2595c8ec31f0c71084b7", path: "crates/editor/src/git/permalink.rs", selection: None, }, ); let expected_url = "https://github.com/zed-industries/zed/blob/e6ebe7974deb6bb6cc0e2595c8ec31f0c71084b7/crates/editor/src/git/permalink.rs"; assert_eq!(permalink.to_string(), expected_url.to_string()) } #[test] fn test_build_github_permalink() { let permalink = Github::public_instance().build_permalink( ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), }, BuildPermalinkParams { sha: "b2efec9824c45fcc90c9a7eb107a50d1772a60aa", path: "crates/zed/src/main.rs", selection: None, }, ); let expected_url = "https://github.com/zed-industries/zed/blob/b2efec9824c45fcc90c9a7eb107a50d1772a60aa/crates/zed/src/main.rs"; assert_eq!(permalink.to_string(), expected_url.to_string()) } #[test] fn test_build_github_permalink_with_single_line_selection() { let permalink = Github::public_instance().build_permalink( ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), }, BuildPermalinkParams { sha: "e6ebe7974deb6bb6cc0e2595c8ec31f0c71084b7", path: "crates/editor/src/git/permalink.rs", selection: Some(6..6), }, ); let expected_url = "https://github.com/zed-industries/zed/blob/e6ebe7974deb6bb6cc0e2595c8ec31f0c71084b7/crates/editor/src/git/permalink.rs#L7"; assert_eq!(permalink.to_string(), expected_url.to_string()) } #[test] fn test_build_github_permalink_with_multi_line_selection() { let permalink = Github::public_instance().build_permalink( ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), }, BuildPermalinkParams { sha: "e6ebe7974deb6bb6cc0e2595c8ec31f0c71084b7", path: "crates/editor/src/git/permalink.rs", selection: Some(23..47), }, ); let expected_url = "https://github.com/zed-industries/zed/blob/e6ebe7974deb6bb6cc0e2595c8ec31f0c71084b7/crates/editor/src/git/permalink.rs#L24-L48"; assert_eq!(permalink.to_string(), expected_url.to_string()) } #[test] fn test_github_pull_requests() { let remote = ParsedGitRemote { owner: "zed-industries".into(), repo: "zed".into(), }; let github = Github::public_instance(); let message = "This does not contain a pull request"; assert!(github.extract_pull_request(&remote, message).is_none()); // Pull request number at end of first line let message = indoc! {r#" project panel: do not expand collapsed worktrees on "collapse all entries" (#10687) Fixes #10597 Release Notes: - Fixed "project panel: collapse all entries" expanding collapsed worktrees. "# }; assert_eq!( github .extract_pull_request(&remote, &message) .unwrap() .url .as_str(), "https://github.com/zed-industries/zed/pull/10687" ); // Pull request number in middle of line, which we want to ignore let message = indoc! {r#" Follow-up to #10687 to fix problems See the original PR, this is a fix. "# }; assert_eq!(github.extract_pull_request(&remote, &message), None); } }