Skip to main content
Deno 2 is finally here 🎉️
Learn more

Downstream

A deno module for downloading files in a streaming fashion. The base of this module is downstream.ts, which queries the file and returns the ReadableStream, as well as a progress Stream in percent (from 0 to 100).

This base function is then augmented with various utility functions for writing this stream into a file or passing it to other streams, as I see fit. (Not included yet)

Simple Usage (without Progress)

const {fileStream, contentLength} = await downstream(File50MB);

// Do anything with the fileStream, what you want, it's a normal WebStream Readable Stream.
// For Example: Write it to a file:

// 1. Create a Temp File
const tempFilePath = await Deno.makeTempFile();

// 2. Open the Temp File for writing
const tempFile = await Deno.open(tempFilePath, {write: true});

// 3. Pipe the file stream (readable stream) into the writeable stream of the tempFile
// CAUTION: The 'await' will block this function until the download has finished!
// For a solution, see next Usage example: 'Extended Usage (with Progress)'
await fileStream.pipeTo(tempFile.writable);

Extended Usage (with ProgressBar Utility)

// 1. Initialize a ProgressBar
const progressBar = new ProgressBar({
    title: 'downloading: ',
    total: 100,
});

// 2. Use the downstream function and get the fileStream and the progressStream out
const {progressStream, fileStream} = await downstream(File1GB);

// 3. Set up handling of your fileStream
// CAUTION: DO NOT use await here, since this will block this function from running,
// until the download has finished!
// Instead,save the promise which is returned from the pipeTo() operator and await it after the progressStream handling,
// at the end of this function.
// If you don't do it like this,
//  your progress stream will not show anything (since deno is waiting for the file handling to finish)

// 3. a) Generate a temp file
const tempFilePath = await Deno.makeTempFile();
const tempFile = await Deno.open(tempFilePath, {write: true});
const fileHandlingPromise = fileStream.pipeTo(tempFile.writable);

// 4. Get each progress stream event and render the state to the progress bar
for await (const progress of progressStream) {
    progressBar.render(Number.parseFloat(progress));
}

// 5. Await the file handling promise from step 3
await fileHandlingPromise;

// 6. Cleanup the temp file
await Deno.remove(tempFilePath);

Running examples

see tasks property in deno.json Run each key there with deno task <task-key>