lazy-file
is a lazy, streaming Blob
/File
implementation for JavaScript.
It allows you to easily create Blob and File objects that defer reading their contents until needed, which is ideal for situations where a file's contents do not fit in memory all at once. When file contents are read, they are streamed to avoid buffering.
- Deferred loading of blob/file contents to minimize memory usage
LazyBlob extends Blob
andLazyFile extends File
so instances can be used anywhere you'd normally expect a regularBlob
/File
- Accepts all the same content types as the original
Blob()
andFile()
constructors - Supports
Blob.slice()
, even on streaming content
JavaScript's File API is useful, but it's not a great fit for streaming server environments where you don't want to buffer file contents. In particular, the File() constructor
requires the contents of a file to be supplied up front when the object is first created, like this:
let file = new File(['hello world'], 'hello.txt', { type: 'text/plain' });
A LazyFile
improves this model by accepting an additional content type in its constructor: LazyContent
.
let lazyContent: LazyContent = {
/* See below for usage */
};
let file = new LazyFile(lazyContent, 'hello.txt', { type: 'text/plain' });
All other File
functionality works as you'd expect.
Install from npm:
npm install @mjackson/lazy-file
The low-level API can be used to create a File
that streams content from anywhere:
import { type LazyContent, LazyFile } from '@mjackson/lazy-file';
let content: LazyContent = {
// The total length of this file in bytes.
byteLength: 100000,
// A function that provides a stream of data for the file contents,
// beginning at the `start` index and ending at `end`.
stream(start, end) {
// ... read the file contents from somewhere and return a ReadableStream
return new ReadableStream({
start(controller) {
controller.enqueue('X'.repeat(100000).slice(start, end));
controller.close();
},
});
},
};
let file = new LazyFile(content, 'example.txt', { type: 'text/plain' });
await file.arrayBuffer(); // ArrayBuffer of the file's content
file.name; // "example.txt"
file.type; // "text/plain"
The lazy-file/fs
export provides functions for reading from and writing to the local filesystem using the File
API.
import { openFile, writeFile } from '@mjackson/lazy-file/fs';
// No data is read at this point, it's just a reference to a
// file on the local filesystem
let file = openFile('./path/to/file.json');
// Data is read when you call file.text() (or any of the
// other Blob methods, like file.bytes(), file.stream(), etc.)
let json = JSON.parse(await file.text());
// Write the file's contents back to the filesystem at a
// different path
await writeFile('./path/to/other-file.json', file);
// Or write to an open file handle/descriptor
import * as fsp from 'node:fs/promises';
let handle = await fsp.open('./path/to/other-file.json');
await writeFile(handle, file);
let imageFile = openFile('./path/to/image.jpg');
// Get a LazyBlob that omits the first 100 bytes of the file.
// This could be useful e.g. when serving HTTP Range requests
let blob = imageFile.slice(100);
All file contents are read on-demand and nothing is ever buffered.
file-storage
- Useslazy-file/fs
internally to create streamingFile
objects from storage on disk
See LICENSE