1use git2::{FetchOptions, Repository};
82use log::debug;
83
84use crate::{
85 error::{PrError, Result},
86 get_remote_callbacks,
87};
88
89#[derive(Debug, Clone, PartialEq, Eq)]
91pub struct PullRequest {
92 pub number: u32,
94 pub remote: Option<String>,
96}
97
98#[derive(Debug, Clone)]
100pub struct PrMetadata {
101 pub number: u32,
103 pub title: String,
105 pub author: String,
107 pub head_ref: String,
109 pub base_ref: String,
111 pub is_fork: bool,
113 pub fork_owner: Option<String>,
115 pub fork_url: Option<String>,
117}
118
119pub fn parse_pr_reference(input: &str) -> Result<Option<PullRequest>> {
131 if let Some(num_str) = input.strip_prefix('#') {
133 return parse_number(num_str, input).map(|num| {
134 Some(PullRequest {
135 number: num,
136 remote: None,
137 })
138 });
139 }
140
141 if let Some(num_str) = input.strip_prefix("pr#") {
143 return parse_number(num_str, input).map(|num| {
144 Some(PullRequest {
145 number: num,
146 remote: None,
147 })
148 });
149 }
150
151 if let Some(num_str) = input.strip_prefix("pr-") {
153 return parse_number(num_str, input).map(|num| {
154 Some(PullRequest {
155 number: num,
156 remote: None,
157 })
158 });
159 }
160
161 if input.contains("github.com") && input.contains("/pull/") {
163 return parse_github_url(input);
164 }
165
166 if input.contains("/pull/") && input.ends_with("/head") {
168 return parse_remote_ref(input);
169 }
170
171 Ok(None)
173}
174
175fn parse_number(num_str: &str, original_input: &str) -> Result<u32> {
177 num_str.parse::<u32>().map_err(|_| {
178 PrError::InvalidReference {
179 input: original_input.to_string(),
180 }
181 .into()
182 })
183}
184
185fn parse_github_url(url: &str) -> Result<Option<PullRequest>> {
187 let parts: Vec<&str> = url.split('/').collect();
189
190 for (i, &part) in parts.iter().enumerate() {
192 if part == "pull" && i + 1 < parts.len() {
193 let num_str = parts[i + 1];
194 let number = parse_number(num_str, url)?;
195 return Ok(Some(PullRequest {
196 number,
197 remote: None,
198 }));
199 }
200 }
201
202 Err(PrError::InvalidReference {
203 input: url.to_string(),
204 }
205 .into())
206}
207
208fn parse_remote_ref(ref_str: &str) -> Result<Option<PullRequest>> {
210 let parts: Vec<&str> = ref_str.split('/').collect();
212
213 if parts.len() >= 4 && parts[parts.len() - 3] == "pull" && parts[parts.len() - 1] == "head" {
214 let num_str = parts[parts.len() - 2];
215 let number = parse_number(num_str, ref_str)?;
216 return Ok(Some(PullRequest {
217 number,
218 remote: None,
219 }));
220 }
221
222 Err(PrError::InvalidReference {
223 input: ref_str.to_string(),
224 }
225 .into())
226}
227
228pub fn check_gh_available() -> Result<()> {
232 std::process::Command::new("gh")
233 .arg("--version")
234 .output()
235 .map_err(|_| PrError::GhNotInstalled)?;
236 Ok(())
237}
238
239pub fn fetch_pr_metadata(pr_number: u32) -> Result<PrMetadata> {
244 check_gh_available()?;
246
247 let output = std::process::Command::new("gh")
249 .args([
250 "pr",
251 "view",
252 &pr_number.to_string(),
253 "--json",
254 "number,title,author,headRefName,baseRefName,isCrossRepository,headRepository",
255 ])
256 .output()
257 .map_err(|e| PrError::GhFetchFailed {
258 message: e.to_string(),
259 })?;
260
261 if !output.status.success() {
262 let stderr = String::from_utf8_lossy(&output.stderr);
263 return Err(PrError::GhFetchFailed {
264 message: stderr.to_string(),
265 }
266 .into());
267 }
268
269 let json_str = String::from_utf8_lossy(&output.stdout);
271 let json: serde_json::Value =
272 serde_json::from_str(&json_str).map_err(|e| PrError::GhJsonParseFailed {
273 message: e.to_string(),
274 })?;
275
276 let number = json["number"]
278 .as_u64()
279 .ok_or_else(|| PrError::GhJsonParseFailed {
280 message: "Missing 'number' field".to_string(),
281 })? as u32;
282
283 let title = json["title"]
284 .as_str()
285 .ok_or_else(|| PrError::GhJsonParseFailed {
286 message: "Missing 'title' field".to_string(),
287 })?
288 .to_string();
289
290 let author = json["author"]["login"]
291 .as_str()
292 .ok_or_else(|| PrError::GhJsonParseFailed {
293 message: "Missing 'author.login' field".to_string(),
294 })?
295 .to_string();
296
297 let head_ref = json["headRefName"]
298 .as_str()
299 .ok_or_else(|| PrError::GhJsonParseFailed {
300 message: "Missing 'headRefName' field".to_string(),
301 })?
302 .to_string();
303
304 let base_ref = json["baseRefName"]
305 .as_str()
306 .ok_or_else(|| PrError::GhJsonParseFailed {
307 message: "Missing 'baseRefName' field".to_string(),
308 })?
309 .to_string();
310
311 let is_fork = json["isCrossRepository"].as_bool().unwrap_or(false);
312
313 let (fork_owner, fork_url) = if is_fork {
314 let owner = json["headRepository"]["owner"]["login"]
315 .as_str()
316 .ok_or(PrError::MissingForkOwner)?
317 .to_string();
318 let url = json["headRepository"]["url"]
319 .as_str()
320 .map(|s| s.to_string());
321 (Some(owner), url)
322 } else {
323 (None, None)
324 };
325
326 Ok(PrMetadata {
327 number,
328 title,
329 author,
330 head_ref,
331 base_ref,
332 is_fork,
333 fork_owner,
334 fork_url,
335 })
336}
337
338fn sanitize_for_branch_name(s: &str) -> String {
340 let sanitized = s
341 .chars()
342 .map(|c| match c {
343 'a'..='z' | 'A'..='Z' | '0'..='9' | '-' | '_' => c,
344 ' ' | '/' => '-',
345 _ => '-',
346 })
347 .collect::<String>()
348 .to_lowercase();
349
350 let mut result = String::new();
352 let mut last_was_dash = false;
353 for c in sanitized.chars() {
354 if c == '-' {
355 if !last_was_dash {
356 result.push(c);
357 }
358 last_was_dash = true;
359 } else {
360 result.push(c);
361 last_was_dash = false;
362 }
363 }
364
365 result.trim_matches(|c| c == '-' || c == '_').to_string()
366}
367
368pub fn format_pr_name_with_metadata(format: &str, metadata: &PrMetadata) -> String {
373 format
374 .replace("{number}", &metadata.number.to_string())
375 .replace("{title}", &sanitize_for_branch_name(&metadata.title))
376 .replace("{author}", &sanitize_for_branch_name(&metadata.author))
377 .replace("{branch}", &sanitize_for_branch_name(&metadata.head_ref))
378}
379
380pub fn is_pr_reference(input: &str) -> bool {
384 parse_pr_reference(input).ok().flatten().is_some()
385}
386
387pub fn detect_pr_remote(repo: &Repository) -> Result<String> {
392 let remotes = repo.remotes()?;
393
394 for name in &["upstream", "origin"] {
396 if remotes.iter().flatten().any(|r| r == *name) {
397 debug!("Using remote: {}", name);
398 return Ok(name.to_string());
399 }
400 }
401
402 if let Some(first_remote) = remotes.get(0) {
404 Ok(first_remote.to_string())
405 } else {
406 Err(PrError::NoRemoteConfigured.into())
407 }
408}
409
410pub fn setup_fork_remote(repo: &Repository, metadata: &PrMetadata) -> Result<String> {
416 if !metadata.is_fork {
417 return detect_pr_remote(repo);
419 }
420
421 let _fork_owner = metadata
423 .fork_owner
424 .as_ref()
425 .ok_or(PrError::MissingForkOwner)?;
426
427 let fork_url = metadata
428 .fork_url
429 .as_ref()
430 .ok_or(PrError::MissingForkOwner)?;
431
432 let fork_remote_name = format!("pr-{}-fork", metadata.number);
434
435 if repo.find_remote(&fork_remote_name).is_ok() {
436 debug!("Fork remote {} already exists", fork_remote_name);
437 return Ok(fork_remote_name);
438 }
439
440 debug!("Adding fork remote: {} -> {}", fork_remote_name, fork_url);
442 repo.remote(&fork_remote_name, fork_url)
443 .map_err(|e| PrError::FetchFailed {
444 remote: fork_remote_name.clone(),
445 message: format!("Failed to add fork remote: {}", e),
446 })?;
447
448 Ok(fork_remote_name)
449}
450
451pub fn fetch_branch(repo: &Repository, remote_name: &str, branch: &str) -> Result<()> {
458 let branch_ref = format!("refs/remotes/{}/{}", remote_name, branch);
460 if repo.find_reference(&branch_ref).is_ok() {
461 debug!("Branch ref {} already exists", branch_ref);
462 return Ok(());
463 }
464
465 debug!("Fetching branch {} from remote {}", branch, remote_name);
466
467 let refspec = format!(
468 "+refs/heads/{}:refs/remotes/{}/{}",
469 branch, remote_name, branch
470 );
471
472 let remote_url = repo
473 .find_remote(remote_name)
474 .ok()
475 .and_then(|r| r.url().map(str::to_string));
476 let mut fetch_options = FetchOptions::new();
477 fetch_options.remote_callbacks(get_remote_callbacks(remote_url.as_deref())?);
478
479 repo.find_remote(remote_name)?
480 .fetch(
481 &[refspec.as_str()],
482 Some(&mut fetch_options),
483 Some("Fetching PR branch"),
484 )
485 .map_err(|e| PrError::FetchFailed {
486 remote: remote_name.to_string(),
487 message: e.message().to_string(),
488 })?;
489
490 debug!("Successfully fetched branch {}", branch);
491 Ok(())
492}
493
494pub fn format_pr_name(format: &str, pr_number: u32) -> String {
498 format.replace("{number}", &pr_number.to_string())
499}
500
501pub fn prepare_pr_worktree(
512 repo: &Repository,
513 pr_number: u32,
514 pr_format: &str,
515) -> Result<(String, String, String)> {
516 debug!("Preparing PR worktree for PR #{}", pr_number);
517
518 let metadata = fetch_pr_metadata(pr_number)?;
520 debug!(
521 "Fetched metadata: title='{}', author='{}', is_fork={}",
522 metadata.title, metadata.author, metadata.is_fork
523 );
524
525 let remote_name = if metadata.is_fork {
529 setup_fork_remote(repo, &metadata)?
530 } else {
531 detect_pr_remote(repo)?
532 };
533
534 fetch_branch(repo, &remote_name, &metadata.head_ref)?;
536
537 let worktree_name = format_pr_name_with_metadata(pr_format, &metadata);
539 debug!("Worktree name: {}", worktree_name);
540
541 let remote_ref = format!("{}/{}", remote_name, metadata.head_ref);
543 debug!("Remote ref: {}", remote_ref);
544
545 Ok((worktree_name, remote_ref, metadata.base_ref))
546}
547
548#[cfg(test)]
549mod tests {
550 use super::*;
551
552 #[test]
553 fn test_parse_hash_number() {
554 let pr = parse_pr_reference("#123").unwrap().unwrap();
555 assert_eq!(pr.number, 123);
556 assert_eq!(pr.remote, None);
557 }
558
559 #[test]
560 fn test_parse_pr_hash_number() {
561 let pr = parse_pr_reference("pr#456").unwrap().unwrap();
562 assert_eq!(pr.number, 456);
563 assert_eq!(pr.remote, None);
564 }
565
566 #[test]
567 fn test_parse_pr_dash_number() {
568 let pr = parse_pr_reference("pr-789").unwrap().unwrap();
569 assert_eq!(pr.number, 789);
570 assert_eq!(pr.remote, None);
571 }
572
573 #[test]
574 fn test_parse_github_url() {
575 let pr = parse_pr_reference("https://github.com/owner/repo/pull/999")
576 .unwrap()
577 .unwrap();
578 assert_eq!(pr.number, 999);
579 assert_eq!(pr.remote, None);
580 }
581
582 #[test]
583 fn test_parse_remote_ref() {
584 let pr = parse_pr_reference("origin/pull/111/head").unwrap().unwrap();
585 assert_eq!(pr.number, 111);
586 assert_eq!(pr.remote, None);
587 }
588
589 #[test]
590 fn test_parse_regular_branch_name() {
591 let result = parse_pr_reference("my-feature-branch").unwrap();
592 assert!(result.is_none());
593 }
594
595 #[test]
596 fn test_parse_invalid_number() {
597 let result = parse_pr_reference("#abc");
598 assert!(result.is_err());
599 }
600
601 #[test]
602 fn test_is_pr_reference_true() {
603 assert!(is_pr_reference("#123"));
604 assert!(is_pr_reference("pr#456"));
605 assert!(is_pr_reference("pr-789"));
606 assert!(is_pr_reference("https://github.com/owner/repo/pull/999"));
607 }
608
609 #[test]
610 fn test_is_pr_reference_false() {
611 assert!(!is_pr_reference("my-branch"));
612 assert!(!is_pr_reference("feature"));
613 }
614
615 #[test]
616 fn test_format_pr_name() {
617 assert_eq!(format_pr_name("pr-{number}", 123), "pr-123");
618 assert_eq!(format_pr_name("review-{number}", 456), "review-456");
619 assert_eq!(format_pr_name("{number}-test", 789), "789-test");
620 }
621
622 #[test]
623 fn test_sanitize_branch_name() {
624 assert_eq!(sanitize_for_branch_name("Fix Bug #123"), "fix-bug-123");
625 assert_eq!(
626 sanitize_for_branch_name("Add Feature (v2)"),
627 "add-feature-v2"
628 );
629 assert_eq!(sanitize_for_branch_name("john-smith"), "john-smith");
630 assert_eq!(
631 sanitize_for_branch_name("Fix: Authentication Issue"),
632 "fix-authentication-issue"
633 );
634 assert_eq!(sanitize_for_branch_name("Test@#$%"), "test");
635 }
636
637 #[test]
638 fn test_format_with_metadata() {
639 let metadata = PrMetadata {
640 number: 123,
641 title: "Fix Authentication Bug".to_string(),
642 author: "john-smith".to_string(),
643 head_ref: "feature/fix-auth".to_string(),
644 base_ref: "main".to_string(),
645 is_fork: false,
646 fork_owner: None,
647 fork_url: None,
648 };
649
650 assert_eq!(
651 format_pr_name_with_metadata("pr-{number}", &metadata),
652 "pr-123"
653 );
654 assert_eq!(
655 format_pr_name_with_metadata("{number}-{title}", &metadata),
656 "123-fix-authentication-bug"
657 );
658 assert_eq!(
659 format_pr_name_with_metadata("{author}/pr-{number}", &metadata),
660 "john-smith/pr-123"
661 );
662 assert_eq!(
663 format_pr_name_with_metadata("{branch}-{number}", &metadata),
664 "feature-fix-auth-123"
665 );
666 }
667
668 #[test]
670 #[ignore]
671 fn test_gh_cli_available() {
672 check_gh_available().expect("gh CLI should be installed");
673 }
674
675 #[test]
676 #[ignore]
677 fn test_fetch_real_pr_metadata() {
678 let metadata = fetch_pr_metadata(1).expect("Failed to fetch PR metadata");
682 assert_eq!(metadata.number, 1);
683 assert!(!metadata.title.is_empty());
684 assert!(!metadata.author.is_empty());
685 }
686}