Duplex streams are a fundamental category of streams in Node.js. However, they’re often misunderstood, including the Duplex stream.
This type of stream is hybrid, meaning it expects a read
and write
method while it’s implemented.
In this article, we’ll review Duplex streams and how to create one. But before that, let’s refresh ourselves on what streams are.
For this post, knowledge of JavaScript and Node.js can help but it is not necessary.
Streams are data collected from a source and brought to another location in a sequence. Streaming a video online is an example: while the video content is passed to you in a sequence, the full content is not available yet.
Streams are divided into four categories: Writable, Readable, Duplex, and Transform.
Readable streams read data from a file or source and pass it to the main application. A buffer then stores the data in case there is a delay passing the data to the application.
When Writable streams, the functionality is opposite. The data is read from the application to the file. There is also a buffer if the data transfer slows, and it then stores it there.
Duplex streams, on the other hand, are a mixture of both the readable and writable streams where both streams are independent of each other.
Transform streams are also like Duplex, but both the readable and writable streams are connected.
The connection enables the application to write data to the application, but there the data must be manipulated before passing to the readable stream.
Just as we explained earlier, the Duplex stream is basically a mixture of the Readable and Writable streams.
An example of a Duplex stream is a Socket, which provides two channels to send and receive data.
Other examples of the Duplex streams are:
To create a Duplex stream in Node.js, begin importing the required methods from the stream module:
const { PassThrough } = require('stream') const tunnel = new PassThrough()
The PassThrough
stream is a basic type of Duplex stream that acts as a tunnel to pipe our Readable stream to the Writable stream.
So, with this tunnel, we can check the data processing to the Writable stream.
Next, let’s read a file using the Readable stream and write it into a Writable stream using writeStream
now:
const { PassThrough } = require("stream"); const { createReadStream, createWriteStream } = require("fs"); const readStream = createReadStream("./README.md"); // read data from this file const writeStream = createWriteStream("./copy.txt"); // write data to this file
Next, we can check what is in the buffer to see if the data is passing in the tunnel:
const { PassThrough } = require("stream"); const { createReadStream, createWriteStream } = require("fs"); const readStream = createReadStream("./README.md"); const writeStream = createWriteStream("./copy.txt"); const tunnel = new PassThrough(); tunnel.on("data", (chunk) => { console.log("bytes:", chunk); // bytes: <Buffer 23 20 4a 61 76 61 53 63 72 69 70 74 20 41 6c 67 6f 72 69 74 68 6d 73 20 61 6e 64 20 44 61 74 61 20 53 74 72 75 63 74 75 72 65 73 0a 0a 54 68 69 73 20 ... 1767 more bytes> }); readStream.pipe(tunnel).pipe(writeStream);
Besides PassThrough
, we have Throttle
to delay how long data passes from one source to another in the pipeline. We can use Duplex streams to set a delay of when the data is brought into our application:
const { PassThrough, Duplex } = require("stream"); const { createReadStream, createWriteStream } = require("fs"); const readStream = createReadStream("./movie.mp4"); const writeStream = createWriteStream("./copy.mp4"); class Throttle extends Duplex { /* * Class constructor will receive the injections as parameters. */ constructor(time) { super(); this.delay = time; } _read() {} // Writes the data, push and set the delay/timeout _write(chunk, encoding, callback) { this.push(chunk); setTimeout(callback, this.delay); } // When all the data is done passing, it stops. _final() { this.push(null); } } const tunnel = new PassThrough(); const throttle = new Throttle(500); let amount = 0; tunnel.on("data", (chunk) => { amount += chunk.length; console.log("bytes:", amount); }); readStream.pipe(throttle).pipe(tunnel).pipe(writeStream);
With this code above, we created a Duplex stream that creates throttle(delay)
for our piped data. This sets a delay of 500 milliseconds, and the method within the Throttle
class pushes the chunk.
The _final()
method only pushes null when the data transfer completes.
We also modified our PassThrough
stream to add up the length of every chunk it reads.
This is how the response renders in the terminal in increments of 500 milliseconds:
By working with Duplex streams in Node.js, we saw how we can delay passing data from one stream to another.
Duplex streams are quite important in our digital world and are used most of the time without us knowing, especially in Socket. These are powerful features because of how they implement both the Readable and Writable streams together.
Deploying a Node-based web app or website is the easy part. Making sure your Node instance continues to serve resources to your app is where things get tougher. If you’re interested in ensuring requests to the backend or third-party services are successful, try LogRocket.
LogRocket is like a DVR for web and mobile apps, recording literally everything that happens while a user interacts with your app. Instead of guessing why problems happen, you can aggregate and report on problematic network requests to quickly understand the root cause.
LogRocket instruments your app to record baseline performance timings such as page load time, time to first byte, slow network requests, and also logs Redux, NgRx, and Vuex actions/state. Start monitoring for free.
Would you be interested in joining LogRocket's developer community?
Join LogRocket’s Content Advisory Board. You’ll help inform the type of content we create and get access to exclusive meetups, social accreditation, and swag.
Sign up nowToast notifications are messages that appear on the screen to provide feedback to users. When users interact with the user […]
Deno’s features and built-in TypeScript support make it appealing for developers seeking a secure and streamlined development experience.
It can be difficult to choose between types and interfaces in TypeScript, but in this post, you’ll learn which to use in specific use cases.
This tutorial demonstrates how to build, integrate, and customize a bottom navigation bar in a Flutter app.