Differential Revision: https://phabricator.services.mozilla.com/D171007
20 KiB
IOUtils Migration Guide
Improving performance through a new file API
What is IOUtils?
IOUtils
is a privileged JavaScript API for performing file I/O in the Firefox frontend.
It was developed as a replacement for OS.File
, addressing
bug 1231711.
It is not to be confused with the unprivileged
DOM File API.
IOUtils
provides a minimal API surface to perform common
I/O tasks via a collection of static methods inspired from OS.File
.
It is implemented in C++, and exposed to JavaScript via WebIDL bindings.
The most up-to-date API can always be found in IOUtils.webidl.
Differences from OS.File
IOUtils
has a similar API to OS.File
, but one should keep in mind some key differences.
No File
instances (except SyncReadFile
in workers)
Most of the IOUtils
methods only operate on absolute path strings, and don't expose a file handle to the caller.
The exception to this rule is the openFileForSyncReading
API, which is only available in workers.
Furthermore, OS.File
was exposing platform-specific file descriptors through the
fd
attribute. IOUtils
does not expose file descriptors.
WebIDL has no Date
type
IOUtils
is written in C++ and exposed to JavaScript through WebIDL.
Many uses of OS.File
concern themselves with obtaining or manipulating file metadata,
like the last modified time, however the Date
type does not exist in WebIDL.
Using IOUtils
,
these values are returned to the caller as the number of milliseconds since
1970-01-01T00:00:00Z
.
Date
s can be safely constructed from these values if needed.
For example, to obtain the last modification time of a file and update it to the current time:
let { lastModified } = await IOUtils.stat(path);
let lastModifiedDate = new Date(lastModified);
let now = new Date();
await IOUtils.touch(path, now.valueOf());
Some methods are not implemented
For various reasons
(complexity, safety, availability of underlying system calls, usefulness, etc.)
the following OS.File
methods have no analogue in IOUtils.
They also will not be implemented.
- void unixSymlink(in string targetPath, in string createPath)
- string getCurrentDirectory(void)
- void setCurrentDirectory(in string path)
- object open(in string path)
- object openUnique(in string path)
Errors are reported as DOMException
s
When an OS.File
method runs into an error,
it will throw/reject with a custom
OS.File.Error
.
These objects have custom attributes that can be checked for common error cases.
IOUtils
has similar behaviour, however its methods consistently reject with a
DOMException
whose name depends on the failure:
Exception Name | Reason for exception |
---|---|
NotFoundError |
A file at the specified path could not be found on disk. |
NotAllowedError |
Access to a file at the specified path was denied by the operating system. |
NotReadableError |
A file at the specified path could not be read for some reason. It may have been too big to read, or it was corrupt, or some other reason. The exception message should have more details. |
ReadOnlyError |
A file at the specified path is read only and could not be modified. |
NoModificationAllowedError |
A file already exists at the specified path and could not be overwritten according to the specified options. The exception message should have more details. |
OperationError |
Something went wrong during the I/O operation. E.g. failed to allocate a buffer. The exception message should have more details. |
UnknownError |
An unknown error occurred in the implementation. An nsresult error code should be included in the exception message to assist with debugging and improving IOUtils internal error handling. |
IOUtils
is mostly async-only
OS.File
provided an asynchronous front-end for main-thread consumers,
and a synchronous front-end for workers.
IOUtils
only provides an asynchronous API for the vast majority of its API surface.
These asynchronous methods can be called from both the main thread and from chrome-privileged worker threads.
The one exception to this rule is openFileForSyncReading
, which allows synchronous file reading in workers.
OS.File
vs IOUtils
Some methods and options of OS.File
keep the same name and underlying behaviour in IOUtils
,
but others have been renamed.
The following is a detailed comparison with examples of the methods and options in each API.
Reading a file
IOUtils
provides the following methods to read data from a file. Like
OS.File
, they accept an options
dictionary.
Note: The maximum file size that can be read is UINT32_MAX
bytes. Attempting
to read a file larger will result in a NotReadableError
.
Promise<Uint8Array> read(DOMString path, ...);
Promise<DOMString> readUTF8(DOMString path, ...);
Promise<any> readJSON(DOMString path, ...);
// Workers only:
SyncReadFile openFileForSyncReading(DOMString path);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
bytes: number? | maxBytes: number? | If specified, read only up to this number of bytes. Otherwise, read the entire file. Default is null. |
compression: 'lz4' | decompress: boolean | If true, read the file and return the decompressed LZ4 stream. Otherwise, just read the file byte-for-byte. Default is false. |
encoding: 'utf-8' | N/A; use readUTF8 instead. |
Interprets the file as UTF-8 encoded text, and returns a string to the caller. |
Examples
Read raw (unsigned) byte values
OS.File
let bytes = await OS.File.read(path); // Uint8Array
IOUtils
let bytes = await IOUtils.read(path); // Uint8Array
Read UTF-8 encoded text
OS.File
let utf8 = await OS.File.read(path, { encoding: 'utf-8' }); // string
IOUtils
let utf8 = await IOUtils.readUTF8(path); // string
Read JSON file
IOUtils
let obj = await IOUtils.readJSON(path); // object
Read LZ4 compressed file contents
OS.File
// Uint8Array
let bytes = await OS.File.read(path, { compression: 'lz4' });
// string
let utf8 = await OS.File.read(path, {
encoding: 'utf-8',
compression: 'lz4',
});
IOUtils
let bytes = await IOUtils.read(path, { decompress: true }); // Uint8Array
let utf8 = await IOUtils.readUTF8(path, { decompress: true }); // string
Synchronously read a fragment of a file into a buffer, from a worker
OS.File
// Read 64 bytes at offset 128, workers only:
let file = OS.File.open(path, { read: true });
file.setPosition(128);
let bytes = file.read({ bytes: 64 }); // Uint8Array
file.close();
IOUtils
// Read 64 bytes at offset 128, workers only:
let file = IOUtils.openFileForSyncReading(path);
let bytes = new Uint8Array(64);
file.readBytesInto(bytes, 128);
file.close();
Writing to a file
IOUtils provides the following methods to write data to a file. Like OS.File, they accept an options dictionary.
Promise<unsigned long long> write(DOMString path, Uint8Array data, ...);
Promise<unsigned long long> writeUTF8(DOMString path, DOMString string, ...);
Promise<unsigned long long> writeJSON(DOMString path, any value, ...);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
backupTo: string? | backupFile: string? | Identifies the path to backup the target file to before performing the write operation. If unspecified, no backup will be performed. Default is null. |
tmpPath: string? | tmpPath: string? | Identifies a path to write to first, before performing a move to overwrite the target file. If unspecified, the target file will be written to directly. Default is null. |
noOverwrite: boolean | mode: 'overwrite' or 'create' | 'create' mode will refuse to overwrite an existing file. Default is 'overwrite'. |
flush: boolean | flush: boolean | If true, force the OS to flush its internal buffers to disk. Default is false. |
encoding: 'utf-8' | N/A; use writeUTF8 instead. |
Allows the caller to supply a string to be encoded as utf-8 text on disk. |
Examples
Write raw (unsigned) byte values
OS.File
let bytes = new Uint8Array();
await OS.File.writeAtomic(path, bytes);
IOUtils
let bytes = new Uint8Array();
await IOUtils.write(path, bytes);
Write UTF-8 encoded text
OS.File
let str = "";
await OS.File.writeAtomic(path, str, { encoding: 'utf-8' });
IOUtils
let str = "";
await IOUtils.writeUTF8(path, str);
Write A JSON object
IOUtils
let obj = {};
await IOUtils.writeJSON(path, obj);
Write with LZ4 compression
OS.File
let bytes = new Uint8Array();
await OS.File.writeAtomic(path, bytes, { compression: 'lz4' });
let str = "";
await OS.File.writeAtomic(path, str, {
compression: 'lz4',
});
IOUtils
let bytes = new Uint8Array();
await IOUtils.write(path, bytes, { compress: true });
let str = "";
await IOUtils.writeUTF8(path, str, { compress: true });
Move a file
IOUtils
provides the following method to move files on disk.
Like OS.File
, it accepts an options dictionary.
Promise<void> move(DOMString sourcePath, DOMString destPath, ...);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
noOverwrite: boolean | noOverwrite: boolean | If true, fail if the destination already exists. Default is false. |
noCopy: boolean | N/A; will not be implemented | This option is not implemented in IOUtils , and will be ignored if provided |
Example
OS.File
await OS.File.move(srcPath, destPath);
IOUtils
await IOUtils.move(srcPath, destPath);
Remove a file
IOUtils
provides one method to remove files from disk.
OS.File
provides several methods.
Promise<void> remove(DOMString path, ...);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
ignoreAbsent: boolean | ignoreAbsent: boolean | If true, and the destination does not exist, then do not raise an error. Default is true. |
N/A; OS.File has dedicated methods for directory removal |
recursive: boolean | If true, and the target is a directory, recursively remove the directory and all its children. Default is false. |
Examples
Remove a file
OS.File
await OS.File.remove(path, { ignoreAbsent: true });
IOUtils
await IOUtils.remove(path);
Remove a directory and all its contents
OS.File
await OS.File.removeDir(path, { ignoreAbsent: true });
IOUtils
await IOUtils.remove(path, { recursive: true });
Remove an empty directory
OS.File
await OS.File.removeEmptyDir(path); // Will throw an exception if `path` is not empty.
IOUtils
await IOUtils.remove(path); // Will throw an exception if `path` is not empty.
Make a directory
IOUtils
provides the following method to create directories on disk.
Like OS.File
, it accepts an options dictionary.
Promise<void> makeDirectory(DOMString path, ...);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
ignoreExisting: boolean | ignoreExisting: boolean | If true, succeed even if the target directory already exists. Default is true. |
from: string | createAncestors: boolean | If true, IOUtils will create all missing ancestors in a path. Default is true. This option differs from OS.File , which requires the caller to specify a root path from which to create missing directories. |
unixMode: number | permissions: unsigned long | The file mode to create the directory with. Ignored on Windows. Default is 0755. |
winSecurity | N/A | IOUtils does not support setting custom directory security settings on Windows. |
Example
OS.File
await OS.File.makeDir(srcPath, destPath);
IOUtils
await IOUtils.makeDirectory(srcPath, destPath);
Update a file's modification time
IOUtils
provides the following method to update a file's modification time.
Promise<void> setModificationTime(DOMString path, optional long long modification);
Example
OS.File
await OS.File.setDates(path, new Date(), new Date());
IOUtils
await IOUtils.setModificationTime(path, new Date().valueOf());
Get file metadata
IOUtils
provides the following method to query file metadata.
Promise<void> stat(DOMString path);
Example
OS.File
let fileInfo = await OS.File.stat(path);
IOUtils
let fileInfo = await IOUtils.stat(path);
Copy a file
IOUtils
provides the following method to copy a file on disk.
Like OS.File
, it accepts an options dictionary.
Promise<void> copy(DOMString path, ...);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
noOverwrite: boolean | noOverwrite: boolean | If true, fail if the destination already exists. Default is false. |
N/A; OS.File does not appear to support recursively copying files |
recursive: boolean | If true, copy the source recursively. |
Examples
Copy a file
OS.File
await OS.File.copy(srcPath, destPath);
IOUtils
await IOUtils.copy(srcPath, destPath);
Copy a directory recursively
OS.File
// Not easy to do.
IOUtils
await IOUtils.copy(srcPath, destPath, { recursive: true });
Iterate a directory
At the moment, IOUtils
does not have a way to expose an iterator for directories.
This is blocked by
bug 1577383.
As a stop-gap for this functionality,
one can get all the children of a directory and iterate through the returned path array using the following method.
Promise<sequence<DOMString>> getChildren(DOMString path);
Example
OS.File
for await (const { path } of new OS.FileDirectoryIterator(dirName)) {</p>
...
}
IOUtils
for (const path of await IOUtils.getChildren(dirName)) {
...
}
Check if a file exists
IOUtils
provides the following method analogous to the OS.File
method of the same name.
Promise<boolean> exists(DOMString path);
Example
OS.File
if (await OS.File.exists(path)) {
...
}
IOUtils
if (await IOUtils.exists(path)) {
...
}
Set the permissions of a file
IOUtils
provides the following method analogous to the OS.File
method of the same name.
Promise<void> setPermissions(DOMString path, unsigned long permissions, optional boolean honorUmask = true);
Options
OS.File option |
IOUtils option |
Description |
---|---|---|
unixMode: number | permissions: unsigned long | The UNIX file mode representing the permissions. Required in IOUtils. |
unixHonorUmask: boolean | honorUmask: boolean | If omitted or true, any UNIX file mode is modified by the permissions. Otherwise the exact value of the permissions will be applied. |
Example
OS.File
await OS.File.setPermissions(path, { unixMode: 0o600 });
IOUtils
await IOUtils.setPermissions(path, 0o600);
FAQs
Why should I use IOUtils
instead of OS.File
?
Bug 1231711 provides some good context, but some reasons include:
- reduced cache-contention,
- faster startup, and
- less memory usage.
Additionally, IOUtils
benefits from a native implementation,
which assists in performance-related work for
Project Fission.
We are actively working to migrate old code usages of OS.File
to analogous IOUtils
calls, so new usages of OS.File
should not be introduced at this time.
Do I need to import anything to use this API?
Nope! It's available via the IOUtils
global in JavaScript (ChromeOnly
context).
Can I use this API from C++ or Rust?
Currently usage is geared exclusively towards JavaScript callers, and all C++ methods are private except for the Web IDL bindings. However given sufficient interest, it should be easy to expose ergonomic public methods for C++ and/or Rust.
Why isn't IOUtils
written in Rust?
At the time of writing, support for Web IDL bindings was more mature for C++ oriented tooling than it was for Rust.
Is IOUtils
feature complete? When will it be available?
IOUtils
is considered feature complete as of Firefox 83.