Path
Path
Represents a path to a file or directory on the filesystem.
DirEntry
Record which represents a directory
This is the same as
Dir.DirEntry
.
IOErr
Tag union of possible errors when reading and writing a file or directory.
This is the same as
File.Err
.
write! :
val,
Path,
fmt
=> Result {} [FileWriteErr Path IOErr]
where val implements Encoding, fmt implements EncoderFormatting
Write data to a file.
First encode a val
using a given fmt
which implements the ability Encode.EncoderFormatting.
For example, suppose you have a Json.toCompactUtf8
which implements
Encode.EncoderFormatting.
You can use this to write JSON
data to a file like this:
# Writes `{"some":"json stuff"}` to the file `output.json`: Path.write! { some: "json stuff" } (Path.from_str "output.json") Json.toCompactUtf8
This opens the file first and closes it after writing to it. If writing to the file fails, for example because of a file permissions issue, the task fails with [WriteErr].
To write unformatted bytes to a file, you can use
Path.write_bytes!
instead.
write_bytes! : List U8, Path => Result {} [FileWriteErr Path IOErr]
Writes bytes to a file.
# Writes the bytes 1, 2, 3 to the file `myfile.dat`. Path.write_bytes! [1, 2, 3] (Path.from_str "myfile.dat")
This opens the file first and closes it after writing to it.
To format data before writing it to a file, you can use
Path.write!
instead.
write_utf8! : Str, Path => Result {} [FileWriteErr Path IOErr]
Writes a Str
to a file, encoded as UTF-8.
# Writes "Hello!" encoded as UTF-8 to the file `myfile.txt`. Path.write_utf8! "Hello!" (Path.from_str "myfile.txt")
This opens the file first and closes it after writing to it.
To write unformatted bytes to a file, you can use
Path.write_bytes!
instead.
from_str : Str -> Path
Note that the path may not be valid depending on the filesystem where it is used.
For example, paths containing :
are valid on ext4 and NTFS filesystems, but not
on FAT ones. So if you have multiple disks on the same machine, but they have
different filesystems, then this path could be valid on one but invalid on another!
It's safest to assume paths are invalid (even syntactically) until given to an operation which uses them to open a file. If that operation succeeds, then the path was valid (at the time). Otherwise, error handling can happen for that operation rather than validating up front for a false sense of security (given symlinks, parts of a path being renamed, etc.).
from_bytes : List U8 -> Path
Not all filesystems use Unicode paths. This function can be used to create a path which
is not valid Unicode (like a Str
is), but which is valid for a particular filesystem.
Note that if the list contains any 0
bytes, sending this path to any file operations
(e.g. Path.read_bytes
or WriteStream.openPath
) will fail.
display : Path -> Str
Unfortunately, operating system paths do not include information about which charset
they were originally encoded with. It's most common (but not guaranteed) that they will
have been encoded with the same charset as the operating system's curent locale (which
typically does not change after it is set during installation of the OS), so
this should convert a Path
to a valid string as long as the path was created
with the given Charset
. (Use Env.charset
to get the current system charset.)
For a conversion to Str
that is lossy but does not return a Result
, see
display
.
toInner : Path -> Str Str, Bytes (List U8)
Assumes a path is encoded as UTF-8,
and converts it to a string using Str.display
.
This conversion is lossy because the path may contain invalid UTF-8 bytes. If that happens,
any invalid bytes will be replaced with the Unicode replacement character
instead of returning an error. As such, it's rarely a good idea to use the Str
returned
by this function for any purpose other than displaying it to a user.
When you don't know for sure what a path's encoding is, UTF-8 is a popular guess because it's the default on UNIX and also is the encoding used in Roc strings. This platform also automatically runs applications under the UTF-8 code page on Windows.
Converting paths to strings can be an unreliable operation, because operating systems
don't record the paths' encodings. This means it's possible for the path to have been
encoded with a different character set than UTF-8 even if UTF-8 is the system default,
which means when display
converts them to a string, the string may include gibberish.
Here is an example.
If you happen to know the Charset
that was used to encode the path, you can use
toStrUsingCharset
instead of display
.
is_dir! : Path => Result Bool [PathErr IOErr]
Returns true if the path exists on disk and is pointing at a directory.
Returns Task.ok false
if the path exists and it is not a directory. If the path does not exist,
this function will return Task.err PathErr PathDoesNotExist
.
This uses rust's std::path::is_dir.
File.is_dir
does the same thing, except it takes aStr
instead of aPath
.
is_file! : Path => Result Bool [PathErr IOErr]
Returns true if the path exists on disk and is pointing at a regular file.
Returns Task.ok false
if the path exists and it is not a file. If the path does not exist,
this function will return Task.err PathErr PathDoesNotExist
.
This uses rust's std::path::is_file.
File.is_file
does the same thing, except it takes aStr
instead of aPath
.
is_sym_link! : Path => Result Bool [PathErr IOErr]
Returns true if the path exists on disk and is pointing at a symbolic link.
Returns Task.ok false
if the path exists and it is not a symbolic link. If the path does not exist,
this function will return Task.err PathErr PathDoesNotExist
.
This uses rust's std::path::is_symlink.
File.is_sym_link
does the same thing, except it takes aStr
instead of aPath
.
type! :
Path
=> Result
[
IsFile,
IsDir,
IsSymLink
] [PathErr IOErr]
Return the type of the path if the path exists on disk.
File.type
does the same thing, except it takes aStr
instead of aPath
.
with_extension : Path, Str -> Path
If the last component of this path has no .
, appends .
followed by the given string.
Otherwise, replaces everything after the last .
with the given string.
# Each of these gives "foo/bar/baz.txt" Path.from_str "foo/bar/baz" |> Path.with_extension "txt" Path.from_str "foo/bar/baz." |> Path.with_extension "txt" Path.from_str "foo/bar/baz.xz" |> Path.with_extension "txt"
delete! : Path => Result {} [FileWriteErr Path IOErr]
Deletes a file from the filesystem.
Performs a DeleteFile
on Windows and unlink
on
UNIX systems. On Windows, this will fail when attempting to delete a readonly
file; the file's readonly permission must be disabled before it can be
successfully deleted.
# Deletes the file named `myfile.dat` Path.delete (Path.from_str "myfile.dat") [1, 2, 3]
This does not securely erase the file's contents from disk; instead, the operating system marks the space it was occupying as safe to write over in the future. Also, the operating system may not immediately mark the space as free; for example, on Windows it will wait until the last file handle to it is closed, and on UNIX, it will not remove it until the last hard link to it has been deleted.
File.delete
does the same thing, except it takes aStr
instead of aPath
.
read_utf8! :
Path
=> Result Str
[
FileReadErr Path IOErr,
FileReadUtf8Err Path
]
Reads a Str
from a file containing UTF-8-encoded text.
# Reads UTF-8 encoded text into a Str from the file "myfile.txt" Path.read_utf8 (Path.from_str "myfile.txt")
This opens the file first and closes it after writing to it.
The task will fail with FileReadUtf8Err
if the given file contains invalid UTF-8.
To read unformatted bytes from a file, you can use
Path.read_bytes!
instead.
File.read_utf8
does the same thing, except it takes aStr
instead of aPath
.
read_bytes! : Path => Result (List U8) [FileReadErr Path IOErr]
Reads all the bytes in a file.
# Read all the bytes in `myfile.txt`. Path.read_bytes! (Path.from_str "myfile.txt")
This opens the file first and closes it after reading its contents.
To read and decode data from a file, you can use
Path.read
instead.
File.read_bytes
does the same thing, except it takes aStr
instead of aPath
.
list_dir! : Path => Result (List Path) [DirErr IOErr]
Lists the files and directories inside the directory.
Dir.list
does the same thing, except it takes aStr
instead of aPath
.
delete_empty! : Path => Result {} [DirErr IOErr]
Deletes a directory if it's empty
This may fail if:
- the path doesn't exist
- the path is not a directory
- the directory is not empty
- the user lacks permission to remove the directory.
Dir.delete_empty
does the same thing, except it takes aStr
instead of aPath
.
delete_all! : Path => Result {} [DirErr IOErr]
Recursively deletes a directory as well as all files and directories inside it.
This may fail if:
- the path doesn't exist
- the path is not a directory
- the directory is not empty
- the user lacks permission to remove the directory.
Dir.delete_all
does the same thing, except it takes aStr
instead of aPath
.
create_dir! : Path => Result {} [DirErr IOErr]
Creates a directory
This may fail if:
- a parent directory does not exist
- the user lacks permission to create a directory there
- the path already exists.
Dir.create
does the same thing, except it takes aStr
instead of aPath
.
create_all! : Path => Result {} [DirErr IOErr]
Creates a directory recursively adding any missing parent directories.
This may fail if:
- the user lacks permission to create a directory there
- the path already exists
Dir.create_all
does the same thing, except it takes aStr
instead of aPath
.
hard_link! : Path => Result {} [LinkErr IOErr]
Creates a new hard link on the filesystem.
The link path will be a link pointing to the original path. Note that systems often require these two paths to both be located on the same filesystem.
This uses rust's std::fs::hard_link.
File.hard_link!
does the same thing, except it takes aStr
instead of aPath
.