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. Dates 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 DOMExceptions

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 noOverwrite: boolean If true, fail if the destination already exists. Default is false.
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 N/A IOUtils does not support setting a custom directory mode on unix.
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: usigned 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.