use std::path::{Path, PathBuf}; use serde::{Deserialize, Serialize}; lazy_static::lazy_static! { pub static ref HOME: PathBuf = dirs::home_dir().expect("failed to determine home directory"); pub static ref CONFIG_DIR: PathBuf = HOME.join(".config").join("zed"); pub static ref CONVERSATIONS_DIR: PathBuf = HOME.join(".config/zed/conversations"); pub static ref LOGS_DIR: PathBuf = HOME.join("Library/Logs/Zed"); pub static ref SUPPORT_DIR: PathBuf = HOME.join("Library/Application Support/Zed"); pub static ref LANGUAGES_DIR: PathBuf = HOME.join("Library/Application Support/Zed/languages"); pub static ref COPILOT_DIR: PathBuf = HOME.join("Library/Application Support/Zed/copilot"); pub static ref DB_DIR: PathBuf = HOME.join("Library/Application Support/Zed/db"); pub static ref SETTINGS: PathBuf = CONFIG_DIR.join("settings.json"); pub static ref KEYMAP: PathBuf = CONFIG_DIR.join("keymap.json"); pub static ref LAST_USERNAME: PathBuf = CONFIG_DIR.join("last-username.txt"); pub static ref LOG: PathBuf = LOGS_DIR.join("Zed.log"); pub static ref OLD_LOG: PathBuf = LOGS_DIR.join("Zed.log.old"); pub static ref LOCAL_SETTINGS_RELATIVE_PATH: &'static Path = Path::new(".zed/settings.json"); } pub mod legacy { use std::path::PathBuf; lazy_static::lazy_static! { static ref CONFIG_DIR: PathBuf = super::HOME.join(".zed"); pub static ref SETTINGS: PathBuf = CONFIG_DIR.join("settings.json"); pub static ref KEYMAP: PathBuf = CONFIG_DIR.join("keymap.json"); } } /// Compacts a given file path by replacing the user's home directory /// prefix with a tilde (`~`). /// /// # Arguments /// /// * `path` - A reference to a `Path` representing the file path to compact. /// /// # Examples /// /// ``` /// use std::path::{Path, PathBuf}; /// use util::paths::compact; /// let path: PathBuf = [ /// util::paths::HOME.to_string_lossy().to_string(), /// "some_file.txt".to_string(), /// ] /// .iter() /// .collect(); /// if cfg!(target_os = "linux") || cfg!(target_os = "macos") { /// assert_eq!(compact(&path).to_str(), Some("~/some_file.txt")); /// } else { /// assert_eq!(compact(&path).to_str(), path.to_str()); /// } /// ``` /// /// # Returns /// /// * A `PathBuf` containing the compacted file path. If the input path /// does not have the user's home directory prefix, or if we are not on /// Linux or macOS, the original path is returned unchanged. pub fn compact(path: &Path) -> PathBuf { if cfg!(target_os = "linux") || cfg!(target_os = "macos") { match path.strip_prefix(HOME.as_path()) { Ok(relative_path) => { let mut shortened_path = PathBuf::new(); shortened_path.push("~"); shortened_path.push(relative_path); shortened_path } Err(_) => path.to_path_buf(), } } else { path.to_path_buf() } } /// A delimiter to use in `path_query:row_number:column_number` strings parsing. pub const FILE_ROW_COLUMN_DELIMITER: char = ':'; /// A representation of a path-like string with optional row and column numbers. /// Matching values example: `te`, `test.rs:22`, `te:22:5`, etc. #[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] pub struct PathLikeWithPosition

{ pub path_like: P, pub row: Option, // Absent if row is absent. pub column: Option, } impl

PathLikeWithPosition

{ /// Parses a string that possibly has `:row:column` suffix. /// Ignores trailing `:`s, so `test.rs:22:` is parsed as `test.rs:22`. /// If any of the row/column component parsing fails, the whole string is then parsed as a path like. pub fn parse_str( s: &str, parse_path_like_str: impl Fn(&str) -> Result, ) -> Result { let fallback = |fallback_str| { Ok(Self { path_like: parse_path_like_str(fallback_str)?, row: None, column: None, }) }; match s.trim().split_once(FILE_ROW_COLUMN_DELIMITER) { Some((path_like_str, maybe_row_and_col_str)) => { let path_like_str = path_like_str.trim(); let maybe_row_and_col_str = maybe_row_and_col_str.trim(); if path_like_str.is_empty() { fallback(s) } else if maybe_row_and_col_str.is_empty() { fallback(path_like_str) } else { let (row_parse_result, maybe_col_str) = match maybe_row_and_col_str.split_once(FILE_ROW_COLUMN_DELIMITER) { Some((maybe_row_str, maybe_col_str)) => { (maybe_row_str.parse::(), maybe_col_str.trim()) } None => (maybe_row_and_col_str.parse::(), ""), }; match row_parse_result { Ok(row) => { if maybe_col_str.is_empty() { Ok(Self { path_like: parse_path_like_str(path_like_str)?, row: Some(row), column: None, }) } else { match maybe_col_str.parse::() { Ok(col) => Ok(Self { path_like: parse_path_like_str(path_like_str)?, row: Some(row), column: Some(col), }), Err(_) => fallback(s), } } } Err(_) => fallback(s), } } } None => fallback(s), } } pub fn map_path_like( self, mapping: impl FnOnce(P) -> Result, ) -> Result, E> { Ok(PathLikeWithPosition { path_like: mapping(self.path_like)?, row: self.row, column: self.column, }) } pub fn to_string(&self, path_like_to_string: impl Fn(&P) -> String) -> String { let path_like_string = path_like_to_string(&self.path_like); if let Some(row) = self.row { if let Some(column) = self.column { format!("{path_like_string}:{row}:{column}") } else { format!("{path_like_string}:{row}") } } else { path_like_string } } } #[cfg(test)] mod tests { use super::*; type TestPath = PathLikeWithPosition; fn parse_str(s: &str) -> TestPath { TestPath::parse_str(s, |s| Ok::<_, std::convert::Infallible>(s.to_string())) .expect("infallible") } #[test] fn path_with_position_parsing_positive() { let input_and_expected = [ ( "test_file.rs", PathLikeWithPosition { path_like: "test_file.rs".to_string(), row: None, column: None, }, ), ( "test_file.rs:1", PathLikeWithPosition { path_like: "test_file.rs".to_string(), row: Some(1), column: None, }, ), ( "test_file.rs:1:2", PathLikeWithPosition { path_like: "test_file.rs".to_string(), row: Some(1), column: Some(2), }, ), ]; for (input, expected) in input_and_expected { let actual = parse_str(input); assert_eq!( actual, expected, "For positive case input str '{input}', got a parse mismatch" ); } } #[test] fn path_with_position_parsing_negative() { for input in [ "test_file.rs:a", "test_file.rs:a:b", "test_file.rs::", "test_file.rs::1", "test_file.rs:1::", "test_file.rs::1:2", "test_file.rs:1::2", "test_file.rs:1:2:", "test_file.rs:1:2:3", ] { let actual = parse_str(input); assert_eq!( actual, PathLikeWithPosition { path_like: input.to_string(), row: None, column: None, }, "For negative case input str '{input}', got a parse mismatch" ); } } // Trim off trailing `:`s for otherwise valid input. #[test] fn path_with_position_parsing_special() { let input_and_expected = [ ( "test_file.rs:", PathLikeWithPosition { path_like: "test_file.rs".to_string(), row: None, column: None, }, ), ( "test_file.rs:1:", PathLikeWithPosition { path_like: "test_file.rs".to_string(), row: Some(1), column: None, }, ), ]; for (input, expected) in input_and_expected { let actual = parse_str(input); assert_eq!( actual, expected, "For special case input str '{input}', got a parse mismatch" ); } } }