Sentry Logo Debug Microservices & Distributed Systems

Join my free newsletter

Level up your dev skills and career with curated tips, practical advice, and in-depth tech insights – all delivered straight to your inbox.

4 min read
Up to date

Trevor I. Lasn

Staff Software Engineer & Engineering Manager

Sharp: A High-Performance Image Processing Library for Node.js

Sharp is a high-performance package for resizing and formatting images

Images play a crucial role in building modern, fast, and user-friendly applications. Nearly every application makes use of images in some way or another.

For example, users may want to resize their profile photos, and merchants often need to shape their images to make them as compelling as possible for buyers.

Large images can be frustrating since they take a long time to load and are challenging to store in a database. Fortunately, there’s a way to make images more user-friendly, load faster, and even appear in higher quality.

The typical use case for the Sharp library module is to convert large images in common formats to smaller, web-friendly JPEG, PNG, and WebP images of varying dimensions, according to the official Sharp documentation.

Formats

The Sharp library is versatile in handling a wide range of image formats. It supports reading images in formats such as JPEG, PNG, WebP, TIFF, GIF, and SVG.

For output, Sharp can generate images in JPEG, PNG, WebP, and TIFF formats, as well as provide uncompressed raw pixel data.

Sharp offers flexibility with input and output options, including Streams, Buffer objects, and the filesystem. You can even split a single input stream into multiple processing pipelines and output streams.

Additionally, Sharp can generate Deep Zoom image pyramids, making it ideal for use with “slippy map” tile viewers like OpenSeadragon.

Getting Started


Terminal window
yarn add sharp

Resizing Images

Let’s say we have a large image that takes forever to load. Instead of serving that big image, we should resize it appropriately, drastically reducing the load time of the application.

To resize images, Sharp only requires us to provide a path to the image, the new dimensions, and the output path for the newly resized image.

const sharp = require('sharp');
sharp('input.jpg')
.resize(500, 450)
.toFile('output.jpg', (err, info) => {
console.log(info);
});

This code resizes the original image to 500x450 dimensions.

Formatting Images

Changing the format of images is straightforward—simply chain functions together. The formatting has a .toFormat() function that takes the new format as the argument.

sharp('input.jpg')
.toFormat('png')
.jpeg({ quality: 100 })
.toFile('output.png', (err, info) => {
console.log(info);
});

Notice we increased the quality of the image to 100, which is the maximum quality the image can have.

The quality specifier is an integer ranging from 1 to 100.

Once we run the code, an output.png file should appear in our project.

PNG output is always full-color at 8 or 16 bits per pixel. Indexed PNG input at 1, 2, or 4 bits per pixel is converted to 8 bits per pixel.

Some of these options require the use of a globally-installed libvips compiled with support for libimagequant (GPL), according to the Sharp official documentation.

Storing the Image in a Database

If you’re working with user profiles, you probably want to save the images in a database. Here’s how you can do that:

sharp('input.jpg')
.resize(500, 450)
.toBuffer()
.then(data => {
// store data in the database
})
.catch(err => {
console.error(err);
});

Notice how Sharp returns a promise instead of a callback. We can use that promise to perform actions when the image processing is done.

Async/Await Example

You can also use the async/await syntax instead of promises. Here’s an example:

async function processImage() {
try {
const data = await sharp('input.jpg')
.resize(500, 450)
.toBuffer();
// store data in the database
} catch (err) {
console.error(err);
}
}
processImage();

Rotating Images

Simple operations such as rotating images are easy with Sharp. Just call the rotate function with the number of degrees you want to rotate the image.

sharp('input.jpg')
.rotate(180)
.toFile('output.jpg', (err, info) => {
console.log(info);
});

You can also use the flip function if you want to rotate your image 180 degrees. This will produce the same output as above.


Become a better engineer

Here are engineering resources I've personally vetted and use. They focus on skills you'll actually need to build and scale real projects - the kind of experience that gets you hired or promoted.

Many companies have a fixed annual stipend per engineer (e.g. $2,000) for use towards learning resources. If your company offers this stipend, you can forward them your invoices directly for reimbursement.


This article was originally published on https://www.trevorlasn.com/blog/sharp-high-performance-node-js-image-processing-library. It was written by a human and polished using grammar tools for clarity.

Interested in a partnership? Shoot me an email at hi [at] trevorlasn.com with all relevant information.