Create a static file server with Node.js
A simple static file server
One of the simplest beginner backend projects you can create is a static file server. In its simplest form, a static file server will listen for requests and try to match the requested URL to a file on the local filesystem. Here's a minimal example of that in action:
import { readFile } from 'fs'; import { createServer } from 'http'; createServer((req, res) => { readFile(__dirname + req.url, (err, data) => { if (err) { res.writeHead(404, { 'Content-Type': 'text/html' }); res.end('404: File not found'); } else { res.writeHead(200, { 'Content-Type': 'text/html' }); res.end(data); } }); }).listen(8000);
In this code example, we're using the fs
module to read the file at __dirname + req.url
. If the file doesn't exist, we'll return a 404
error. Otherwise, we'll return the file. The http
module is used to create the server that listens on port 8000
.
In theory, one could stop here and have a very basic static file server. However, there are a few considerations that could be taken into account. Let's explore them one by one, and see how we can address them.
Modularity
First and foremost, we don't necessarily want to serve files from the same directory as our Node.js server. To address this problem, we would have to change the directory fs.readFile()
looks for the file in. To accomplish this, we can specify a directory to serve files from and use the path
module to resolve files from that directory. This way, we can also better handle different operating systems and environments.
Here's a short code snippet on how to resolve a file path using the path
module:
import { readFile } from 'fs'; import { join } from 'path'; const directoryName = './public'; const requestUrl = 'index.html'; const filePath = join(directoryName, requestUrl); readFile(filePath, (err, data) => { // ... });
Security
Our next concern is security. Obviously, we don't want users prying around our machine unauthorized. Currently, it's not impossible to get access to files outside of the specified root directory (e.g. GET /../../../
). To address this, we can use the path
module again to check if the requested file is inside the root directory.
import { join, normalize, resolve } from 'path'; const directoryName = './public'; const root = normalize(resolve(directoryName)); const requestUrl = 'index.html'; const filePath = join(root, fileName); const isPathUnderRoot = normalize(resolve(filePath)).startsWith(root);
Similarly, we can ensure that users don't get access to sensitive files by checking the file type. For this to work, we can specify an array or object of supported file types and check the file's extension using the path
module once again.
import { extname } from 'path'; const types = ['html', 'css', 'js', 'json']; const requestUrl = 'index.html'; const extension = extname(requestUrl).slice(1); const isTypeSupported = types.includes(extension);
Omitting the HTML extension
A staple of most websites is the ability to omit the file extension from the URL when requesting an HTML page. It's a small quality of life improvement that users expect and it would be really nice to add to our static file server.
This is where things get a little tricky. To provide this functionality, we need to check for missing extensions and look up the appropriate HTML file. Bear in mind, though, that there are two possible matches for a URL such as /my-page
. This path can either be matched by /my-page.html
or my-page/index.html
. To deal with this, we'll prioritize one over the other. In our case, we'll prioritize /my-page.html
over my-page/index.html
, but it's pretty easy to swap them the other way round.
To implement this, we can use the fs
module to check if one of them exists and handle things appropriately. A special case would also need to be added for the root url (/
) to match it to the index.html
file.
import { accessSync, constants } from 'fs'; import { join, normalize, resolve, extname } from 'path'; const directoryName = './public'; const root = normalize(resolve(directoryName)); const extension = extname(req.url).slice(1); let fileName = requestUrl; if (requestUrl === '/') fileName = 'index.html'; else if (!extension) { try { accessSync(join(root, requestUrl + '.html'), constants.F_OK); fileName = requestUrl + '.html'; } catch (e) { fileName = join(requestUrl, 'index.html'); } }
Final touches
After implementing all of the above, we can put everything together to create a static file server with all the functionality we need. I'll throw in a couple of finishing touches, such as logging requests to the console and handling a few more file types, and here's the final product:
import { readFile, accessSync, constants } from 'fs'; import { createServer } from 'http'; import { join, normalize, resolve, extname } from 'path'; const port = 8000; const directoryName = './public'; const types = { html: 'text/html', css: 'text/css', js: 'application/javascript', png: 'image/png', jpg: 'image/jpeg', jpeg: 'image/jpeg', gif: 'image/gif', json: 'application/json', xml: 'application/xml', }; const root = normalize(resolve(directoryName)); const server = createServer((req, res) => { console.log(`${req.method} ${req.url}`); const extension = extname(req.url).slice(1); const type = extension ? types[extension] : types.html; const supportedExtension = Boolean(type); if (!supportedExtension) { res.writeHead(404, { 'Content-Type': 'text/html' }); res.end('404: File not found'); return; } let fileName = req.url; if (req.url === '/') fileName = 'index.html'; else if (!extension) { try { accessSync(join(root, req.url + '.html'), constants.F_OK); fileName = req.url + '.html'; } catch (e) { fileName = join(req.url, 'index.html'); } } const filePath = join(root, fileName); const isPathUnderRoot = normalize(resolve(filePath)).startsWith(root); if (!isPathUnderRoot) { res.writeHead(404, { 'Content-Type': 'text/html' }); res.end('404: File not found'); return; } readFile(filePath, (err, data) => { if (err) { res.writeHead(404, { 'Content-Type': 'text/html' }); res.end('404: File not found'); } else { res.writeHead(200, { 'Content-Type': type }); res.end(data); } }); }); server.listen(port, () => { console.log(`Server is listening on port ${port}`); });
Not too bad, right? In just 70 lines of code, we managed to create a pretty decent static file server without using anything but core Node.js APIs.