use std::{ops::Range, path::PathBuf}; #[derive(Debug, Clone, PartialEq)] pub struct PathRange { pub path: PathBuf, pub range: Option>, } #[derive(Debug, Clone, Copy, PartialEq, Eq)] pub struct LineCol { pub line: u32, pub col: Option, } impl LineCol { pub fn new(str: impl AsRef) -> Option { let str = str.as_ref(); match str.split_once(':') { Some((line, col)) => match (line.parse::(), col.parse::()) { (Ok(line), Ok(col)) => Some(Self { line, col: Some(col), }), _ => None, }, None => match str.parse::() { Ok(line) => Some(Self { line, col: None }), Err(_) => None, }, } } } impl PathRange { pub fn new(str: impl AsRef) -> Self { let str = str.as_ref(); // Sometimes the model will include a language at the start, // e.g. "```rust zed/crates/markdown/src/markdown.rs#L1" // We just discard that. let str = match str.trim_end().rfind(' ') { Some(space) => &str[space + 1..], None => str.trim_start(), }; match str.rsplit_once('#') { Some((path, after_hash)) => { // Be tolerant to the model omitting the "L" prefix, lowercasing it, // or including it more than once. let after_hash = after_hash.replace(['L', 'l'], ""); let range = { let mut iter = after_hash.split('-').flat_map(LineCol::new); iter.next() .map(|start| iter.next().map(|end| start..end).unwrap_or(start..start)) }; Self { path: PathBuf::from(path), range, } } None => Self { path: str.into(), range: None, }, } } } #[cfg(test)] mod tests { use super::*; #[test] fn test_linecol_parsing() { let line_col = LineCol::new("10:5"); assert_eq!( line_col, Some(LineCol { line: 10, col: Some(5) }) ); let line_only = LineCol::new("42"); assert_eq!( line_only, Some(LineCol { line: 42, col: None }) ); assert_eq!(LineCol::new(""), None); assert_eq!(LineCol::new("not a number"), None); assert_eq!(LineCol::new("10:not a number"), None); assert_eq!(LineCol::new("not:5"), None); } #[test] fn test_pathrange_parsing() { let path_range = PathRange::new("file.rs#L10-L20"); assert_eq!(path_range.path, PathBuf::from("file.rs")); assert!(path_range.range.is_some()); if let Some(range) = path_range.range { assert_eq!(range.start.line, 10); assert_eq!(range.start.col, None); assert_eq!(range.end.line, 20); assert_eq!(range.end.col, None); } let single_line = PathRange::new("file.rs#L15"); assert_eq!(single_line.path, PathBuf::from("file.rs")); assert!(single_line.range.is_some()); if let Some(range) = single_line.range { assert_eq!(range.start.line, 15); assert_eq!(range.end.line, 15); } let no_range = PathRange::new("file.rs"); assert_eq!(no_range.path, PathBuf::from("file.rs")); assert!(no_range.range.is_none()); let lowercase = PathRange::new("file.rs#l5-l10"); assert_eq!(lowercase.path, PathBuf::from("file.rs")); assert!(lowercase.range.is_some()); if let Some(range) = lowercase.range { assert_eq!(range.start.line, 5); assert_eq!(range.end.line, 10); } let complex = PathRange::new("src/path/to/file.rs#L100"); assert_eq!(complex.path, PathBuf::from("src/path/to/file.rs")); assert!(complex.range.is_some()); } #[test] fn test_pathrange_from_str() { let with_range = PathRange::new("file.rs#L10-L20"); assert!(with_range.range.is_some()); assert_eq!(with_range.path, PathBuf::from("file.rs")); let without_range = PathRange::new("file.rs"); assert!(without_range.range.is_none()); let single_line = PathRange::new("file.rs#L15"); assert!(single_line.range.is_some()); } #[test] fn test_pathrange_leading_text_trimming() { let with_language = PathRange::new("```rust file.rs#L10"); assert_eq!(with_language.path, PathBuf::from("file.rs")); assert!(with_language.range.is_some()); if let Some(range) = with_language.range { assert_eq!(range.start.line, 10); } let with_spaces = PathRange::new("``` file.rs#L10-L20"); assert_eq!(with_spaces.path, PathBuf::from("file.rs")); assert!(with_spaces.range.is_some()); let with_words = PathRange::new("```rust code example file.rs#L15:10"); assert_eq!(with_words.path, PathBuf::from("file.rs")); assert!(with_words.range.is_some()); if let Some(range) = with_words.range { assert_eq!(range.start.line, 15); assert_eq!(range.start.col, Some(10)); } let with_whitespace = PathRange::new(" file.rs#L5"); assert_eq!(with_whitespace.path, PathBuf::from("file.rs")); assert!(with_whitespace.range.is_some()); let no_leading = PathRange::new("file.rs#L10"); assert_eq!(no_leading.path, PathBuf::from("file.rs")); assert!(no_leading.range.is_some()); } #[test] fn test_pathrange_with_line_and_column() { let line_and_col = PathRange::new("file.rs#L10:5"); assert_eq!(line_and_col.path, PathBuf::from("file.rs")); assert!(line_and_col.range.is_some()); if let Some(range) = line_and_col.range { assert_eq!(range.start.line, 10); assert_eq!(range.start.col, Some(5)); assert_eq!(range.end.line, 10); assert_eq!(range.end.col, Some(5)); } let full_range = PathRange::new("file.rs#L10:5-L20:15"); assert_eq!(full_range.path, PathBuf::from("file.rs")); assert!(full_range.range.is_some()); if let Some(range) = full_range.range { assert_eq!(range.start.line, 10); assert_eq!(range.start.col, Some(5)); assert_eq!(range.end.line, 20); assert_eq!(range.end.col, Some(15)); } let mixed_range1 = PathRange::new("file.rs#L10:5-L20"); assert_eq!(mixed_range1.path, PathBuf::from("file.rs")); assert!(mixed_range1.range.is_some()); if let Some(range) = mixed_range1.range { assert_eq!(range.start.line, 10); assert_eq!(range.start.col, Some(5)); assert_eq!(range.end.line, 20); assert_eq!(range.end.col, None); } let mixed_range2 = PathRange::new("file.rs#L10-L20:15"); assert_eq!(mixed_range2.path, PathBuf::from("file.rs")); assert!(mixed_range2.range.is_some()); if let Some(range) = mixed_range2.range { assert_eq!(range.start.line, 10); assert_eq!(range.start.col, None); assert_eq!(range.end.line, 20); assert_eq!(range.end.col, Some(15)); } } }