2011-06-15 13:19:50 -05:00
|
|
|
|
2011-06-17 13:12:51 -05:00
|
|
|
import os::getcwd;
|
2011-06-17 17:52:37 -05:00
|
|
|
import os_fs;
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-10-19 23:53:02 -05:00
|
|
|
native "c-stack-cdecl" mod rustrt {
|
2011-10-04 20:24:30 -05:00
|
|
|
fn rust_file_is_dir(path: str::sbuf) -> int;
|
2011-03-10 08:56:51 -06:00
|
|
|
}
|
|
|
|
|
2011-09-02 17:34:58 -05:00
|
|
|
fn path_sep() -> str { ret str::from_char(os_fs::path_sep); }
|
2011-03-10 08:56:51 -06:00
|
|
|
|
2011-09-02 17:34:58 -05:00
|
|
|
type path = str;
|
2011-03-10 08:56:51 -06:00
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn dirname(p: path) -> path {
|
2011-09-01 19:27:58 -05:00
|
|
|
let i: int = str::rindex(p, os_fs::path_sep as u8);
|
2011-07-27 07:19:39 -05:00
|
|
|
if i == -1 {
|
2011-09-01 19:27:58 -05:00
|
|
|
i = str::rindex(p, os_fs::alt_path_sep as u8);
|
2011-09-02 17:34:58 -05:00
|
|
|
if i == -1 { ret "."; }
|
2011-03-10 08:56:51 -06:00
|
|
|
}
|
2011-09-01 19:27:58 -05:00
|
|
|
ret str::substr(p, 0u, i as uint);
|
2011-03-10 08:56:51 -06:00
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn basename(p: path) -> path {
|
2011-09-01 19:27:58 -05:00
|
|
|
let i: int = str::rindex(p, os_fs::path_sep as u8);
|
2011-07-27 07:19:39 -05:00
|
|
|
if i == -1 {
|
2011-09-01 19:27:58 -05:00
|
|
|
i = str::rindex(p, os_fs::alt_path_sep as u8);
|
2011-07-27 07:19:39 -05:00
|
|
|
if i == -1 { ret p; }
|
2011-06-07 19:54:22 -05:00
|
|
|
}
|
2011-09-01 19:27:58 -05:00
|
|
|
let len = str::byte_len(p);
|
2011-07-27 07:19:39 -05:00
|
|
|
if i + 1 as uint >= len { ret p; }
|
2011-09-01 19:27:58 -05:00
|
|
|
ret str::slice(p, i + 1 as uint, len);
|
2011-06-07 19:54:22 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-05-20 19:47:14 -05:00
|
|
|
// FIXME: Need some typestate to avoid bounds check when len(pre) == 0
|
2011-09-12 04:27:30 -05:00
|
|
|
fn connect(pre: path, post: path) -> path {
|
2011-09-01 19:27:58 -05:00
|
|
|
let len = str::byte_len(pre);
|
2011-08-19 17:16:48 -05:00
|
|
|
ret if pre[len - 1u] == os_fs::path_sep as u8 {
|
2011-07-27 07:19:39 -05:00
|
|
|
|
2011-06-16 18:55:46 -05:00
|
|
|
// Trailing '/'?
|
2011-06-15 13:19:50 -05:00
|
|
|
pre + post
|
|
|
|
} else { pre + path_sep() + post };
|
2011-03-15 19:11:55 -05:00
|
|
|
}
|
|
|
|
|
2011-09-28 19:06:57 -05:00
|
|
|
fn connect_many(paths: [path]) : vec::is_not_empty(paths) -> path {
|
|
|
|
ret if vec::len(paths) == 1u {
|
|
|
|
paths[0]
|
|
|
|
} else {
|
|
|
|
let rest = vec::slice(paths, 1u, vec::len(paths));
|
|
|
|
check vec::is_not_empty(rest);
|
|
|
|
connect(paths[0], connect_many(rest))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn file_is_dir(p: path) -> bool {
|
2011-10-04 20:24:30 -05:00
|
|
|
ret str::as_buf(p, {|buf| rustrt::rust_file_is_dir(buf) != 0 });
|
2011-08-24 18:00:26 -05:00
|
|
|
}
|
2011-03-10 08:56:51 -06:00
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn list_dir(p: path) -> [str] {
|
2011-08-24 18:00:26 -05:00
|
|
|
let p = p;
|
2011-09-01 19:27:58 -05:00
|
|
|
let pl = str::byte_len(p);
|
2011-08-19 17:16:48 -05:00
|
|
|
if pl == 0u || p[pl - 1u] as char != os_fs::path_sep { p += path_sep(); }
|
2011-09-02 17:34:58 -05:00
|
|
|
let full_paths: [str] = [];
|
|
|
|
for filename: str in os_fs::list_dir(p) {
|
|
|
|
if !str::eq(filename, ".") {
|
|
|
|
if !str::eq(filename, "..") { full_paths += [p + filename]; }
|
2011-06-15 13:19:50 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ret full_paths;
|
2011-03-10 08:56:51 -06:00
|
|
|
}
|
2011-06-17 13:12:51 -05:00
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn path_is_absolute(p: path) -> bool { ret os_fs::path_is_absolute(p); }
|
2011-06-17 13:12:51 -05:00
|
|
|
|
2011-06-17 17:52:37 -05:00
|
|
|
// FIXME: under Windows, we should prepend the current drive letter to paths
|
|
|
|
// that start with a slash.
|
2011-09-12 04:27:30 -05:00
|
|
|
fn make_absolute(p: path) -> path {
|
2011-09-02 17:34:58 -05:00
|
|
|
if path_is_absolute(p) { ret p; } else { ret connect(getcwd(), p); }
|
2011-06-17 13:12:51 -05:00
|
|
|
}
|
|
|
|
|
2011-10-04 22:18:04 -05:00
|
|
|
fn split(p: path) -> [path] {
|
|
|
|
let split1 = str::split(p, os_fs::path_sep as u8);
|
|
|
|
let split2 = [];
|
|
|
|
for s in split1 {
|
|
|
|
split2 += str::split(s, os_fs::alt_path_sep as u8);
|
|
|
|
}
|
|
|
|
ret split2;
|
|
|
|
}
|
|
|
|
|
2011-10-04 23:02:37 -05:00
|
|
|
fn normalize(p: path) -> path {
|
|
|
|
let s = split(p);
|
|
|
|
let s = strip_dots(s);
|
|
|
|
let s = rollup_doubledots(s);
|
|
|
|
|
|
|
|
let s = if check vec::is_not_empty(s) {
|
|
|
|
connect_many(s)
|
|
|
|
} else {
|
|
|
|
""
|
|
|
|
};
|
|
|
|
let s = reabsolute(p, s);
|
|
|
|
let s = reterminate(p, s);
|
|
|
|
|
|
|
|
let s = if str::byte_len(s) == 0u {
|
|
|
|
"."
|
|
|
|
} else {
|
|
|
|
s
|
|
|
|
};
|
|
|
|
|
|
|
|
ret s;
|
|
|
|
|
|
|
|
fn strip_dots(s: [path]) -> [path] {
|
|
|
|
vec::filter_map({ |elem|
|
|
|
|
if elem == "." {
|
|
|
|
option::none
|
|
|
|
} else {
|
|
|
|
option::some(elem)
|
|
|
|
}
|
|
|
|
}, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn rollup_doubledots(s: [path]) -> [path] {
|
|
|
|
if vec::is_empty(s) {
|
|
|
|
ret [];
|
|
|
|
}
|
|
|
|
|
|
|
|
let t = [];
|
|
|
|
let i = vec::len(s);
|
|
|
|
let skip = 0;
|
|
|
|
do {
|
|
|
|
i -= 1u;
|
|
|
|
if s[i] == ".." {
|
|
|
|
skip += 1;
|
|
|
|
} else {
|
|
|
|
if skip == 0 {
|
|
|
|
t += [s[i]];
|
|
|
|
} else {
|
|
|
|
skip -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while i != 0u;
|
|
|
|
let t = vec::reversed(t);
|
|
|
|
while skip > 0 {
|
|
|
|
t += [".."];
|
|
|
|
skip -= 1;
|
|
|
|
}
|
|
|
|
ret t;
|
|
|
|
}
|
|
|
|
|
2011-10-05 14:01:10 -05:00
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
#[cfg(target_os = "macos")]
|
2011-10-04 23:02:37 -05:00
|
|
|
fn reabsolute(orig: path, new: path) -> path {
|
|
|
|
if path_is_absolute(orig) {
|
|
|
|
path_sep() + new
|
|
|
|
} else {
|
|
|
|
new
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-05 14:01:10 -05:00
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn reabsolute(orig: path, new: path) -> path {
|
|
|
|
if path_is_absolute(orig) && orig[0] == os_fs::path_sep as u8 {
|
|
|
|
str::from_char(os_fs::path_sep) + new
|
|
|
|
} else {
|
|
|
|
new
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-04 23:02:37 -05:00
|
|
|
fn reterminate(orig: path, new: path) -> path {
|
|
|
|
let last = orig[str::byte_len(orig) - 1u];
|
|
|
|
if last == os_fs::path_sep as u8
|
|
|
|
|| last == os_fs::path_sep as u8 {
|
|
|
|
ret new + path_sep();
|
|
|
|
} else {
|
|
|
|
ret new;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-16 16:58:02 -05:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust;
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
2011-05-05 15:53:57 -05:00
|
|
|
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
|
2011-03-16 16:58:02 -05:00
|
|
|
// End:
|