2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Module: fs
|
|
|
|
|
|
|
|
File system manipulation
|
|
|
|
*/
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-12-18 15:46:04 -06:00
|
|
|
import core::ctypes;
|
2011-12-13 18:25:51 -06:00
|
|
|
import core::vec;
|
|
|
|
import core::option;
|
2011-11-08 22:35:15 -06:00
|
|
|
import os;
|
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-11-16 22:49:38 -06:00
|
|
|
#[abi = "cdecl"]
|
|
|
|
native mod rustrt {
|
2012-01-16 04:21:01 -06:00
|
|
|
fn rust_path_is_dir(path: str::sbuf) -> ctypes::c_int;
|
|
|
|
fn rust_path_exists(path: str::sbuf) -> ctypes::c_int;
|
2011-03-10 08:56:51 -06:00
|
|
|
}
|
|
|
|
|
2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: path_sep
|
|
|
|
|
|
|
|
Get the default path separator for the host platform
|
|
|
|
*/
|
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-10-24 17:25:41 -05:00
|
|
|
// FIXME: This type should probably be constrained
|
|
|
|
/*
|
|
|
|
Type: path
|
|
|
|
|
|
|
|
A path or fragment of a filesystem path
|
|
|
|
*/
|
2011-09-02 17:34:58 -05:00
|
|
|
type path = str;
|
2011-03-10 08:56:51 -06:00
|
|
|
|
2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: dirname
|
|
|
|
|
|
|
|
Get the directory portion of a path
|
|
|
|
|
|
|
|
Returns all of the path up to, but excluding, the final path separator.
|
|
|
|
The dirname of "/usr/share" will be "/usr", but the dirname of
|
|
|
|
"/usr/share/" is "/usr/share".
|
|
|
|
|
|
|
|
If the path is not prefixed with a directory, then "." is returned.
|
|
|
|
*/
|
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-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: basename
|
|
|
|
|
|
|
|
Get the file name portion of a path
|
|
|
|
|
|
|
|
Returns the portion of the path after the final path separator.
|
|
|
|
The basename of "/usr/share" will be "share". If there are no
|
|
|
|
path separators in the path then the returned path is identical to
|
|
|
|
the provided path. If an empty path is provided or the path ends
|
|
|
|
with a path separator then an empty path is returned.
|
|
|
|
*/
|
2012-02-01 05:41:58 -06:00
|
|
|
fn basename(p: path) -> path unsafe {
|
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; }
|
2012-02-01 06:20:21 -06:00
|
|
|
ret str::unsafe::slice_bytes(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-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: connect
|
|
|
|
|
|
|
|
Connects to path segments
|
|
|
|
|
2012-01-09 18:11:25 -06:00
|
|
|
Given paths `pre` and `post, removes any trailing path separator on `pre` and
|
|
|
|
any leading path separator on `post`, and returns the concatenation of the two
|
|
|
|
with a single path separator between them.
|
2011-10-24 17:25:41 -05:00
|
|
|
*/
|
2011-07-27 07:19:39 -05:00
|
|
|
|
2012-01-09 18:11:25 -06:00
|
|
|
fn connect(pre: path, post: path) -> path {
|
|
|
|
let pre_ = pre;
|
|
|
|
let post_ = post;
|
|
|
|
let sep = os_fs::path_sep as u8;
|
|
|
|
let pre_len = str::byte_len(pre);
|
|
|
|
let post_len = str::byte_len(post);
|
|
|
|
if pre_len > 1u && pre[pre_len-1u] == sep { str::pop_byte(pre_); }
|
|
|
|
if post_len > 1u && post[0] == sep { str::shift_byte(post_); }
|
|
|
|
ret pre_ + path_sep() + post_;
|
2011-03-15 19:11:55 -05:00
|
|
|
}
|
|
|
|
|
2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: connect_many
|
|
|
|
|
|
|
|
Connects a vector of path segments into a single path.
|
|
|
|
|
|
|
|
Inserts path separators as needed.
|
|
|
|
*/
|
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-10-24 17:25:41 -05:00
|
|
|
/*
|
2011-12-16 16:32:09 -06:00
|
|
|
Function: path_is_dir
|
2011-10-24 17:25:41 -05:00
|
|
|
|
|
|
|
Indicates whether a path represents a directory.
|
|
|
|
*/
|
2011-12-16 16:32:09 -06:00
|
|
|
fn path_is_dir(p: path) -> bool {
|
2012-01-18 05:29:37 -06:00
|
|
|
ret str::as_buf(p, {|buf|
|
2012-02-05 18:21:52 -06:00
|
|
|
rustrt::rust_path_is_dir(buf) != 0 as ctypes::c_int
|
2012-01-18 05:29:37 -06:00
|
|
|
});
|
2011-12-16 16:32:09 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Function: path_exists
|
|
|
|
|
|
|
|
Indicates whether a path exists.
|
|
|
|
*/
|
|
|
|
fn path_exists(p: path) -> bool {
|
2012-01-18 05:29:37 -06:00
|
|
|
ret str::as_buf(p, {|buf|
|
|
|
|
rustrt::rust_path_exists(buf) != 0 as ctypes::c_int
|
|
|
|
});
|
2011-08-24 18:00:26 -05:00
|
|
|
}
|
2011-03-10 08:56:51 -06:00
|
|
|
|
2011-11-08 22:35:15 -06:00
|
|
|
/*
|
|
|
|
Function: make_dir
|
|
|
|
|
2011-11-22 16:09:35 -06:00
|
|
|
Creates a directory at the specified path.
|
2011-11-08 22:35:15 -06:00
|
|
|
*/
|
2011-11-22 18:11:03 -06:00
|
|
|
fn make_dir(p: path, mode: ctypes::c_int) -> bool {
|
2011-11-08 22:35:15 -06:00
|
|
|
ret mkdir(p, mode);
|
|
|
|
|
|
|
|
#[cfg(target_os = "win32")]
|
2011-11-22 18:11:03 -06:00
|
|
|
fn mkdir(_p: path, _mode: ctypes::c_int) -> bool unsafe {
|
2011-11-08 22:35:15 -06:00
|
|
|
// FIXME: turn mode into something useful?
|
2011-11-08 23:27:48 -06:00
|
|
|
ret str::as_buf(_p, {|buf|
|
2011-11-22 17:14:34 -06:00
|
|
|
os::kernel32::CreateDirectoryA(
|
|
|
|
buf, unsafe::reinterpret_cast(0))
|
2011-11-08 23:27:48 -06:00
|
|
|
});
|
2011-11-08 22:35:15 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
#[cfg(target_os = "macos")]
|
2011-12-30 02:18:55 -06:00
|
|
|
#[cfg(target_os = "freebsd")]
|
2011-11-22 18:11:03 -06:00
|
|
|
fn mkdir(_p: path, _mode: ctypes::c_int) -> bool {
|
|
|
|
ret str::as_buf(_p, {|buf| os::libc::mkdir(buf, _mode) == 0i32 });
|
2011-11-08 22:35:15 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: list_dir
|
|
|
|
|
|
|
|
Lists the contents of a directory.
|
|
|
|
*/
|
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-11-22 16:09:35 -06:00
|
|
|
/*
|
|
|
|
Function: remove_dir
|
|
|
|
|
|
|
|
Removes a directory at the specified path.
|
|
|
|
*/
|
|
|
|
fn remove_dir(p: path) -> bool {
|
|
|
|
ret rmdir(p);
|
|
|
|
|
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn rmdir(_p: path) -> bool {
|
2011-11-22 17:14:34 -06:00
|
|
|
ret str::as_buf(_p, {|buf| os::kernel32::RemoveDirectoryA(buf)});
|
2011-11-22 16:09:35 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
#[cfg(target_os = "macos")]
|
2011-12-30 02:18:55 -06:00
|
|
|
#[cfg(target_os = "freebsd")]
|
2011-11-22 16:09:35 -06:00
|
|
|
fn rmdir(_p: path) -> bool {
|
2011-11-22 18:11:03 -06:00
|
|
|
ret str::as_buf(_p, {|buf| os::libc::rmdir(buf) == 0i32 });
|
2011-11-22 16:09:35 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-01 21:19:10 -06:00
|
|
|
fn change_dir(p: path) -> bool {
|
|
|
|
ret chdir(p);
|
|
|
|
|
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn chdir(_p: path) -> bool {
|
2011-12-01 21:31:12 -06:00
|
|
|
ret str::as_buf(_p, {|buf| os::kernel32::SetCurrentDirectoryA(buf)});
|
2011-12-01 21:19:10 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
#[cfg(target_os = "macos")]
|
2011-12-30 02:18:55 -06:00
|
|
|
#[cfg(target_os = "freebsd")]
|
2011-12-01 21:19:10 -06:00
|
|
|
fn chdir(_p: path) -> bool {
|
|
|
|
ret str::as_buf(_p, {|buf| os::libc::chdir(buf) == 0i32 });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: path_is_absolute
|
|
|
|
|
|
|
|
Indicates whether a path is absolute.
|
|
|
|
|
|
|
|
A path is considered absolute if it begins at the filesystem root ("/") or,
|
|
|
|
on Windows, begins with a drive letter.
|
|
|
|
*/
|
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-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: make_absolute
|
|
|
|
|
|
|
|
Convert a relative path to an absolute path
|
|
|
|
|
|
|
|
If the given path is relative, return it prepended with the current working
|
|
|
|
directory. If the given path is already an absolute path, return it
|
|
|
|
as is.
|
|
|
|
*/
|
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-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: split
|
|
|
|
|
|
|
|
Split a path into it's individual components
|
|
|
|
|
|
|
|
Splits a given path by path separators and returns a vector containing
|
|
|
|
each piece of the path. On Windows, if the path is absolute then
|
|
|
|
the first element of the returned vector will be the drive letter
|
|
|
|
followed by a colon.
|
|
|
|
*/
|
2011-10-04 22:18:04 -05:00
|
|
|
fn split(p: path) -> [path] {
|
2012-02-01 22:31:01 -06:00
|
|
|
// FIXME: use UTF-8 safe str, and/or various other string formats
|
|
|
|
let split1 = str::split_byte(p, os_fs::path_sep as u8);
|
2011-10-04 22:18:04 -05:00
|
|
|
let split2 = [];
|
|
|
|
for s in split1 {
|
2012-02-01 22:31:01 -06:00
|
|
|
split2 += str::split_byte(s, os_fs::alt_path_sep as u8);
|
2011-10-04 22:18:04 -05:00
|
|
|
}
|
2012-02-01 22:31:01 -06:00
|
|
|
|
|
|
|
// filter out ""
|
|
|
|
let split3 = vec::filter(split2, {|seg| "" != seg});
|
|
|
|
ret split3;
|
2011-10-04 22:18:04 -05:00
|
|
|
}
|
|
|
|
|
2011-10-29 01:23:51 -05:00
|
|
|
/*
|
|
|
|
Function: splitext
|
|
|
|
|
|
|
|
Split a path into a pair of strings with the first element being the filename
|
|
|
|
without the extension and the second being either empty or the file extension
|
|
|
|
including the period. Leading periods in the basename are ignored. If the
|
|
|
|
path includes directory components then they are included in the filename part
|
|
|
|
of the result pair.
|
|
|
|
*/
|
|
|
|
fn splitext(p: path) -> (str, str) {
|
2012-02-01 22:31:01 -06:00
|
|
|
// FIXME: use UTF-8 safe str, and/or various other string formats
|
2011-10-29 01:23:51 -05:00
|
|
|
if str::is_empty(p) { ("", "") }
|
|
|
|
else {
|
2012-02-01 22:31:01 -06:00
|
|
|
let parts = str::split_byte(p, '.' as u8);
|
2011-10-29 01:23:51 -05:00
|
|
|
if vec::len(parts) > 1u {
|
|
|
|
let base = str::connect(vec::init(parts), ".");
|
|
|
|
let ext = "." + option::get(vec::last(parts));
|
|
|
|
|
|
|
|
fn is_dotfile(base: str) -> bool {
|
|
|
|
str::is_empty(base)
|
|
|
|
|| str::ends_with(
|
|
|
|
base, str::from_char(os_fs::path_sep))
|
|
|
|
|| str::ends_with(
|
|
|
|
base, str::from_char(os_fs::alt_path_sep))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ext_contains_sep(ext: str) -> bool {
|
|
|
|
vec::len(split(ext)) > 1u
|
|
|
|
}
|
|
|
|
|
|
|
|
fn no_basename(ext: str) -> bool {
|
|
|
|
str::ends_with(
|
|
|
|
ext, str::from_char(os_fs::path_sep))
|
|
|
|
|| str::ends_with(
|
|
|
|
ext, str::from_char(os_fs::alt_path_sep))
|
|
|
|
}
|
|
|
|
|
|
|
|
if is_dotfile(base)
|
|
|
|
|| ext_contains_sep(ext)
|
|
|
|
|| no_basename(ext) {
|
|
|
|
(p, "")
|
|
|
|
} else {
|
|
|
|
(base, ext)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
(p, "")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-24 17:25:41 -05:00
|
|
|
/*
|
|
|
|
Function: normalize
|
|
|
|
|
|
|
|
Removes extra "." and ".." entries from paths.
|
|
|
|
|
|
|
|
Does not follow symbolic links.
|
|
|
|
*/
|
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] {
|
2011-12-16 08:27:50 -06:00
|
|
|
vec::filter_map(s, { |elem|
|
2011-10-04 23:02:37 -05:00
|
|
|
if elem == "." {
|
|
|
|
option::none
|
|
|
|
} else {
|
|
|
|
option::some(elem)
|
|
|
|
}
|
2011-12-16 08:27:50 -06:00
|
|
|
})
|
2011-10-04 23:02:37 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
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-12-30 02:18:55 -06:00
|
|
|
#[cfg(target_os = "freebsd")]
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-06 13:55:43 -06:00
|
|
|
/*
|
|
|
|
Function: homedir
|
|
|
|
|
|
|
|
Returns the path to the user's home directory, if known.
|
|
|
|
|
|
|
|
On Unix, returns the value of the "HOME" environment variable if it is set and
|
|
|
|
not equal to the empty string.
|
|
|
|
|
|
|
|
On Windows, returns the value of the "HOME" environment variable if it is set
|
|
|
|
and not equal to the empty string. Otherwise, returns the value of the
|
|
|
|
"USERPROFILE" environment variable if it is set and not equal to the empty
|
|
|
|
string.
|
|
|
|
|
|
|
|
Otherwise, homedir returns option::none.
|
|
|
|
*/
|
|
|
|
fn homedir() -> option<path> {
|
|
|
|
ret alt generic_os::getenv("HOME") {
|
|
|
|
some(p) {
|
2012-01-06 18:37:04 -06:00
|
|
|
if !str::is_empty(p) {
|
2012-01-06 13:55:43 -06:00
|
|
|
some(p)
|
|
|
|
} else {
|
2012-01-06 18:37:04 -06:00
|
|
|
secondary()
|
|
|
|
}
|
|
|
|
}
|
2012-01-19 00:37:22 -06:00
|
|
|
none {
|
2012-01-06 18:37:04 -06:00
|
|
|
secondary()
|
|
|
|
}
|
2012-01-06 13:55:43 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
#[cfg(target_os = "macos")]
|
|
|
|
#[cfg(target_os = "freebsd")]
|
|
|
|
fn secondary() -> option<path> {
|
|
|
|
none
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn secondary() -> option<path> {
|
|
|
|
option::maybe(none, generic_os::getenv("USERPROFILE")) {|p|
|
|
|
|
if !str::is_empty(p) {
|
|
|
|
some(p)
|
2012-01-06 18:37:04 -06:00
|
|
|
} else {
|
|
|
|
none
|
|
|
|
}
|
2012-01-06 13:55:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-17 21:05:07 -06:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
#[test]
|
|
|
|
fn test_connect() {
|
|
|
|
let slash = fs::path_sep();
|
|
|
|
log(error, fs::connect("a", "b"));
|
|
|
|
assert (fs::connect("a", "b") == "a" + slash + "b");
|
|
|
|
assert (fs::connect("a" + slash, "b") == "a" + slash + "b");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Issue #712
|
|
|
|
#[test]
|
|
|
|
fn test_list_dir_no_invalid_memory_access() { fs::list_dir("."); }
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn list_dir() {
|
|
|
|
let dirs = fs::list_dir(".");
|
|
|
|
// Just assuming that we've got some contents in the current directory
|
|
|
|
assert (vec::len(dirs) > 0u);
|
|
|
|
|
|
|
|
for dir in dirs { log(debug, dir); }
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn path_is_dir() {
|
|
|
|
assert (fs::path_is_dir("."));
|
|
|
|
assert (!fs::path_is_dir("test/stdtest/fs.rs"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn path_exists() {
|
|
|
|
assert (fs::path_exists("."));
|
|
|
|
assert (!fs::path_exists("test/nonexistent-bogus-path"));
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ps() -> str {
|
|
|
|
fs::path_sep()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn aps() -> str {
|
|
|
|
"/"
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn split1() {
|
|
|
|
let actual = fs::split("a" + ps() + "b");
|
|
|
|
let expected = ["a", "b"];
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn split2() {
|
|
|
|
let actual = fs::split("a" + aps() + "b");
|
|
|
|
let expected = ["a", "b"];
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn split3() {
|
|
|
|
let actual = fs::split(ps() + "a" + ps() + "b");
|
|
|
|
let expected = ["a", "b"];
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn split4() {
|
|
|
|
let actual = fs::split("a" + ps() + "b" + aps() + "c");
|
|
|
|
let expected = ["a", "b", "c"];
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize1() {
|
|
|
|
let actual = fs::normalize("a/b/..");
|
|
|
|
let expected = "a";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize2() {
|
|
|
|
let actual = fs::normalize("/a/b/..");
|
|
|
|
let expected = "/a";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize3() {
|
|
|
|
let actual = fs::normalize("a/../b");
|
|
|
|
let expected = "b";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize4() {
|
|
|
|
let actual = fs::normalize("/a/../b");
|
|
|
|
let expected = "/b";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize5() {
|
|
|
|
let actual = fs::normalize("a/.");
|
|
|
|
let expected = "a";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize6() {
|
|
|
|
let actual = fs::normalize("a/./b/");
|
|
|
|
let expected = "a/b/";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize7() {
|
|
|
|
let actual = fs::normalize("a/..");
|
|
|
|
let expected = ".";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize8() {
|
|
|
|
let actual = fs::normalize("../../..");
|
|
|
|
let expected = "../../..";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize9() {
|
|
|
|
let actual = fs::normalize("a/b/../../..");
|
|
|
|
let expected = "..";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize10() {
|
|
|
|
let actual = fs::normalize("/a/b/c/../d/./../../e/");
|
|
|
|
let expected = "/a/e/";
|
|
|
|
log(error, actual);
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn normalize11() {
|
|
|
|
let actual = fs::normalize("/a/..");
|
|
|
|
let expected = "/";
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn normalize12() {
|
|
|
|
let actual = fs::normalize("C:/whatever");
|
|
|
|
let expected = "C:/whatever";
|
|
|
|
log(error, actual);
|
|
|
|
assert actual == expected;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn path_is_absolute_win32() {
|
|
|
|
assert fs::path_is_absolute("C:/whatever");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_empty() {
|
|
|
|
let (base, ext) = fs::splitext("");
|
|
|
|
assert base == "";
|
|
|
|
assert ext == "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_ext() {
|
|
|
|
let (base, ext) = fs::splitext("grum.exe");
|
|
|
|
assert base == "grum";
|
|
|
|
assert ext == ".exe";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_noext() {
|
|
|
|
let (base, ext) = fs::splitext("grum");
|
|
|
|
assert base == "grum";
|
|
|
|
assert ext == "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_dotfile() {
|
|
|
|
let (base, ext) = fs::splitext(".grum");
|
|
|
|
assert base == ".grum";
|
|
|
|
assert ext == "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_path_ext() {
|
|
|
|
let (base, ext) = fs::splitext("oh/grum.exe");
|
|
|
|
assert base == "oh/grum";
|
|
|
|
assert ext == ".exe";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_path_noext() {
|
|
|
|
let (base, ext) = fs::splitext("oh/grum");
|
|
|
|
assert base == "oh/grum";
|
|
|
|
assert ext == "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_dot_in_path() {
|
|
|
|
let (base, ext) = fs::splitext("oh.my/grum");
|
|
|
|
assert base == "oh.my/grum";
|
|
|
|
assert ext == "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn splitext_nobasename() {
|
|
|
|
let (base, ext) = fs::splitext("oh.my/");
|
|
|
|
assert base == "oh.my/";
|
|
|
|
assert ext == "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
#[cfg(target_os = "macos")]
|
|
|
|
#[cfg(target_os = "freebsd")]
|
|
|
|
fn homedir() {
|
|
|
|
import getenv = generic_os::getenv;
|
|
|
|
import setenv = generic_os::setenv;
|
|
|
|
|
|
|
|
let oldhome = getenv("HOME");
|
|
|
|
|
|
|
|
setenv("HOME", "/home/MountainView");
|
|
|
|
assert fs::homedir() == some("/home/MountainView");
|
|
|
|
|
|
|
|
setenv("HOME", "");
|
|
|
|
assert fs::homedir() == none;
|
|
|
|
|
|
|
|
option::may(oldhome, {|s| setenv("HOME", s)});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_os = "win32")]
|
|
|
|
fn homedir() {
|
|
|
|
import getenv = generic_os::getenv;
|
|
|
|
import setenv = generic_os::setenv;
|
|
|
|
|
|
|
|
let oldhome = getenv("HOME");
|
|
|
|
let olduserprofile = getenv("USERPROFILE");
|
|
|
|
|
|
|
|
setenv("HOME", "");
|
|
|
|
setenv("USERPROFILE", "");
|
|
|
|
|
|
|
|
assert fs::homedir() == none;
|
|
|
|
|
|
|
|
setenv("HOME", "/home/MountainView");
|
|
|
|
assert fs::homedir() == some("/home/MountainView");
|
|
|
|
|
|
|
|
setenv("HOME", "");
|
|
|
|
|
|
|
|
setenv("USERPROFILE", "/home/MountainView");
|
|
|
|
assert fs::homedir() == some("/home/MountainView");
|
|
|
|
|
|
|
|
setenv("USERPROFILE", "/home/MountainView");
|
|
|
|
assert fs::homedir() == some("/home/MountainView");
|
|
|
|
|
|
|
|
setenv("HOME", "/home/MountainView");
|
|
|
|
setenv("USERPROFILE", "/home/PaloAlto");
|
|
|
|
assert fs::homedir() == some("/home/MountainView");
|
|
|
|
|
|
|
|
option::may(oldhome, {|s| setenv("HOME", s)});
|
|
|
|
option::may(olduserprofile, {|s| setenv("USERPROFILE", s)});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test() {
|
|
|
|
assert (!fs::path_is_absolute("test-path"));
|
|
|
|
|
|
|
|
log(debug, "Current working directory: " + os::getcwd());
|
|
|
|
|
|
|
|
log(debug, fs::make_absolute("test-path"));
|
|
|
|
log(debug, fs::make_absolute("/usr/bin"));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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
|
|
|
|
// End:
|