1use crate::*;
19
20pub fn build_abs_path(root: &str, path: &str) -> String {
27 debug_assert!(root.starts_with('/'), "root must start with /");
28 debug_assert!(root.ends_with('/'), "root must end with /");
29
30 let p = root[1..].to_string();
31
32 if path == "/" {
33 p
34 } else {
35 debug_assert!(!path.starts_with('/'), "path must not start with /");
36 p + path
37 }
38}
39
40pub fn build_rooted_abs_path(root: &str, path: &str) -> String {
47 debug_assert!(root.starts_with('/'), "root must start with /");
48 debug_assert!(root.ends_with('/'), "root must end with /");
49
50 let p = root.to_string();
51
52 if path == "/" {
53 p
54 } else {
55 debug_assert!(!path.starts_with('/'), "path must not start with /");
56 p + path
57 }
58}
59
60pub fn build_rel_path(root: &str, path: &str) -> String {
68 debug_assert!(root != path, "get rel path with root is invalid");
69
70 if path.starts_with('/') {
71 debug_assert!(
72 path.starts_with(root),
73 "path {path} doesn't start with root {root}"
74 );
75 path[root.len()..].to_string()
76 } else {
77 debug_assert!(
78 path.starts_with(&root[1..]),
79 "path {path} doesn't start with root {root}"
80 );
81 path[root.len() - 1..].to_string()
82 }
83}
84
85pub fn normalize_path(path: &str) -> String {
97 let path = path.trim().trim_start_matches('/');
100
101 if path.is_empty() {
103 return "/".to_string();
104 }
105
106 let has_trailing = path.ends_with('/');
107
108 let mut p = path
109 .split('/')
110 .filter(|v| !v.is_empty())
111 .collect::<Vec<&str>>()
112 .join("/");
113
114 if has_trailing {
116 p.push('/');
117 }
118
119 p
120}
121
122pub fn normalize_root(v: &str) -> String {
135 let mut v = v
136 .split('/')
137 .filter(|v| !v.is_empty())
138 .collect::<Vec<&str>>()
139 .join("/");
140 if !v.starts_with('/') {
141 v.insert(0, '/');
142 }
143 if !v.ends_with('/') {
144 v.push('/')
145 }
146 v
147}
148
149pub fn get_basename(path: &str) -> &str {
151 if path == "/" {
153 return "/";
154 }
155
156 if !path.ends_with('/') {
158 return path
159 .split('/')
160 .next_back()
161 .expect("file path without name is invalid");
162 }
163
164 let idx = path[..path.len() - 1].rfind('/').map(|v| v + 1);
168
169 match idx {
170 Some(v) => {
171 let (_, name) = path.split_at(v);
172 name
173 }
174 None => path,
175 }
176}
177
178pub fn get_parent(path: &str) -> &str {
180 if path == "/" {
181 return "/";
182 }
183
184 if !path.ends_with('/') {
185 let idx = path.rfind('/');
189
190 return match idx {
191 Some(v) => {
192 let (parent, _) = path.split_at(v + 1);
193 parent
194 }
195 None => "/",
196 };
197 }
198
199 let idx = path[..path.len() - 1].rfind('/').map(|v| v + 1);
203
204 match idx {
205 Some(v) => {
206 let (parent, _) = path.split_at(v);
207 parent
208 }
209 None => "/",
210 }
211}
212
213pub fn validate_path(path: &str, mode: EntryMode) -> bool {
215 debug_assert!(!path.is_empty(), "input path should not be empty");
216
217 match mode {
218 EntryMode::FILE => !path.ends_with('/'),
219 EntryMode::DIR => path.ends_with('/'),
220 EntryMode::Unknown => false,
221 }
222}
223
224#[cfg(test)]
225mod tests {
226 use super::*;
227
228 #[test]
229 fn test_normalize_path() {
230 let cases = vec![
231 ("file path", "abc", "abc"),
232 ("dir path", "abc/", "abc/"),
233 ("empty path", "", "/"),
234 ("root path", "/", "/"),
235 ("root path with extra /", "///", "/"),
236 ("abs file path", "/abc/def", "abc/def"),
237 ("abs dir path", "/abc/def/", "abc/def/"),
238 ("abs file path with extra /", "///abc/def", "abc/def"),
239 ("abs dir path with extra /", "///abc/def/", "abc/def/"),
240 ("file path contains ///", "abc///def", "abc/def"),
241 ("dir path contains ///", "abc///def///", "abc/def/"),
242 ("file with whitespace", "abc/def ", "abc/def"),
243 ];
244
245 for (name, input, expect) in cases {
246 assert_eq!(normalize_path(input), expect, "{name}")
247 }
248 }
249
250 #[test]
251 fn test_normalize_root() {
252 let cases = vec![
253 ("dir path", "abc/", "/abc/"),
254 ("empty path", "", "/"),
255 ("root path", "/", "/"),
256 ("root path with extra /", "///", "/"),
257 ("abs dir path", "/abc/def/", "/abc/def/"),
258 ("abs file path with extra /", "///abc/def", "/abc/def/"),
259 ("abs dir path with extra /", "///abc/def/", "/abc/def/"),
260 ("dir path contains ///", "abc///def///", "/abc/def/"),
261 ];
262
263 for (name, input, expect) in cases {
264 assert_eq!(normalize_root(input), expect, "{name}")
265 }
266 }
267
268 #[test]
269 fn test_get_basename() {
270 let cases = vec![
271 ("file abs path", "foo/bar/baz.txt", "baz.txt"),
272 ("file rel path", "bar/baz.txt", "baz.txt"),
273 ("file walk", "foo/bar/baz", "baz"),
274 ("dir rel path", "bar/baz/", "baz/"),
275 ("dir root", "/", "/"),
276 ("dir walk", "foo/bar/baz/", "baz/"),
277 ];
278
279 for (name, input, expect) in cases {
280 let actual = get_basename(input);
281 assert_eq!(actual, expect, "{name}")
282 }
283 }
284
285 #[test]
286 fn test_get_parent() {
287 let cases = vec![
288 ("file abs path", "foo/bar/baz.txt", "foo/bar/"),
289 ("file rel path", "bar/baz.txt", "bar/"),
290 ("file walk", "foo/bar/baz", "foo/bar/"),
291 ("dir rel path", "bar/baz/", "bar/"),
292 ("dir root", "/", "/"),
293 ("dir abs path", "/foo/bar/", "/foo/"),
294 ("dir walk", "foo/bar/baz/", "foo/bar/"),
295 ];
296
297 for (name, input, expect) in cases {
298 let actual = get_parent(input);
299 assert_eq!(actual, expect, "{name}")
300 }
301 }
302
303 #[test]
304 fn test_build_abs_path() {
305 let cases = vec![
306 ("input abs file", "/abc/", "/", "abc/"),
307 ("input dir", "/abc/", "def/", "abc/def/"),
308 ("input file", "/abc/", "def", "abc/def"),
309 ("input abs file with root /", "/", "/", ""),
310 ("input empty with root /", "/", "", ""),
311 ("input dir with root /", "/", "def/", "def/"),
312 ("input file with root /", "/", "def", "def"),
313 ];
314
315 for (name, root, input, expect) in cases {
316 let actual = build_abs_path(root, input);
317 assert_eq!(actual, expect, "{name}")
318 }
319 }
320
321 #[test]
322 fn test_build_rooted_abs_path() {
323 let cases = vec![
324 ("input abs file", "/abc/", "/", "/abc/"),
325 ("input dir", "/abc/", "def/", "/abc/def/"),
326 ("input file", "/abc/", "def", "/abc/def"),
327 ("input abs file with root /", "/", "/", "/"),
328 ("input dir with root /", "/", "def/", "/def/"),
329 ("input file with root /", "/", "def", "/def"),
330 ];
331
332 for (name, root, input, expect) in cases {
333 let actual = build_rooted_abs_path(root, input);
334 assert_eq!(actual, expect, "{name}")
335 }
336 }
337
338 #[test]
339 fn test_build_rel_path() {
340 let cases = vec![
341 ("input abs file", "/abc/", "/abc/def", "def"),
342 ("input dir", "/abc/", "/abc/def/", "def/"),
343 ("input file", "/abc/", "abc/def", "def"),
344 ("input dir with root /", "/", "def/", "def/"),
345 ("input file with root /", "/", "def", "def"),
346 ];
347
348 for (name, root, input, expect) in cases {
349 let actual = build_rel_path(root, input);
350 assert_eq!(actual, expect, "{name}")
351 }
352 }
353
354 #[test]
355 fn test_validate_path() {
356 let cases = vec![
357 ("input file with mode file", "abc", EntryMode::FILE, true),
358 ("input file with mode dir", "abc", EntryMode::DIR, false),
359 ("input dir with mode file", "abc/", EntryMode::FILE, false),
360 ("input dir with mode dir", "abc/", EntryMode::DIR, true),
361 ("root with mode dir", "/", EntryMode::DIR, true),
362 (
363 "input file with mode unknown",
364 "abc",
365 EntryMode::Unknown,
366 false,
367 ),
368 (
369 "input dir with mode unknown",
370 "abc/",
371 EntryMode::Unknown,
372 false,
373 ),
374 ];
375
376 for (name, path, mode, expect) in cases {
377 let actual = validate_path(path, mode);
378 assert_eq!(actual, expect, "{name}")
379 }
380 }
381}