express-autoindex
v1.4.2
Published
express-autoindex produce a directory listing like Nginx, Apache or another, but just with express
Downloads
60
Maintainers
Readme
express-autoindex
express-autoindex
produce a directory listing like Nginx, Apache or another, but just with express
It takes into consideration most of the mime-types correctly, and page generation is fully customizable.
The objectives are:
- Make a
HTML
page orJSON
data easily usable, on the great majority of browsers - Correctly take into consideration the majority of
MIME types
, and the generation of the most customizable pages possible - Supports most common file encodings
- Native support for
Typescript
,EcmaScript
andCommonJS
- The least amount of dependency possible (currently only
two
) - The
lightest
possible
- Install
- API
- Customization of the html page appearance
- Customization of the date format
- Customization of the JSON format
- Error handling
- Minimalist example
- Production mode
- To do list
- License
- Dependencies
Install
# npm
npm install express-autoindex
# yarn
yarn add express-autoindex
API
import autoindex from 'express-autoindex';
// Root of server, ./public dir
app.use(autoindex('public'))
// Specific path `/files`, ./public dir
app.use('/files', autoindex('public'));
// Set options
app.use('/files', autoindex('public', { dirAtTop: false, displaySize: false }));
autoindex(path, options)
Returns middlware that serves an index of the directory in the given path
.
The path
is based of the req.url
value, so a req.url
of '/some/dir
with a path
of 'public'
will look at 'public/some/dir'
.
Options
express-autoindex
accepts options:
alwaysThrowError
type:
boolean
Throw error for all HTTP error codes (4xx & 5xx).
By default, errors will be generated only on 5xx types. If you wish to generate an error regardless of the HTTP error code, pass
true
to the option.Default to
false
cache
type:
number | false
Caches for a defined time the generated pages. Very useful to save server resources.
Pass
false
to disable the cache, or the number of milliseconds representing the cache expiration time.Default to
300000
= 5 minscustomJsonFormat
type:
object
{ isDir: string, name: string, path: string, time: string, size: string }
By default, the json generated for a file or folder follows a precise structure. It is possible to rename or remove the key of this object.
More detail here
Default to
undefined
customTemplate
type:
string
Pass the relative path of your custom template file. For example, if the file is located in the same folder of your startup server file, simply write
my-file.html
or./my-file.html
.More detail here
Default to
undefined
dateFormat
type:
string
Custom date print format.
More detail here
Default to
undefined
dirAtTop
type:
boolean
Display directories before files
Default to
true
displayDate
type:
boolean
Display the last modification date of the file or directory if available.
Default to
true
displayDotfile
type:
boolean
Display dotfiles (.env, .yarnrc, ...).
Default to
false
displaySize
type:
boolean
Display size of the file or directory if available.
Default to
true
exclude
type:
RegExp
Regular expression for files/dirs exclude, for example
/my-file.json|\*.cpp/
.json
type:
boolean
Send data in json format instead of an html page. Might be useful if you want to use the data for another application.
Default to
false
strict
type:
boolean
Allow only
HEAD
andGET
HTTP methods.Default to
true
Customization of the html page appearance
It is possible to customize the entire HTML page sent to the client. To do this, write an HTML page as usual. Then simply pass the path to your file to the customTemplate
option.
Variables
express-autoindex
generates two variables:
- title : the title of the generated page. Since this is an autoindex, the title represents the folder path
- content : the contents of an html table representing the contents of the folder
To use variables in your template, simply call them between two curly brackets, like {{title}}
.
CSS
Elements inside a row of table all have their own css class linked to it, for easy access via a css selector. These classes are all placed on a :
- link : url pointing to a file or folder
- size : folder or file size
- time : date of last folder or file modification
Don't forget that by default, every browser provides its own native css. If you want to standardize the look and feel, a little css will be necessary.
HTML example
The html code below is the one generated by default. You can use it as inspiration to generate your own template.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{title}}</title>
</head>
<body>
<h1>{{title}}</h1>
<hr/>
<table>{{content}}</table>
<hr/>
</body>
<style type="text/css">
html {
font-family: Arial, Helvetica, sans-serif
}
table {
font-family: 'Courier New', Courier, monospace;
font-size: 12px;
font-weight: 400;
letter-spacing: normal;
line-height: normal;
font-style: normal;
}
tr td:first-child {
min-width: 20%;
}
td a {
margin-right: 1em;
}
td.size {
text-align: end;
}
</style>
</html>
Customization of the date format
It is possible to customize the entire date format of file or folder. To do this, pass a string containing the new format to the dateFormat
option.
The date is a string
containing the desired keys, by default %d?-%mo-%y %h:%mi
.
It is formatted in UTC format, and has eight keys:
- %wd → week day
- %d → day
- %mo → month
- %y → year
- %h → hours
- %mi → minutes
- %s → seconds
- %ms → milliseconds
Optional character
It's possible for one of these keys to return an empty string
if nothing is available.
Now let's imagine that a separator is just after it. If it appears when the key is empty, it's not aesthetically pleasing.
You can therefore add the ? character just after a key. This super character will ensure that the one following it only appears if the key is not empty.
For example, with the following format %d?-%y
, if %d is empty, the string returned will be 2023
and not -2023
.
Date format example
US: %mo-%d?-$y %h:%mi
EU: %d?-%mo-%y %h:%mi
ISO: %y-%mo-%dT%h:%mi:%sZ
Custom format: <%d?-%y ~ %h>
Customization of the JSON format
It is possible to customize the entire JSON format of file or folder. To do this, pass an object containing the new key names to the customJsonFormat
option.
By default, the object generated for an element is:
- isDir → (boolean) is a folder
- name → (string) the name
- path → (string) the path (url)
- time → (string) the UTC date of the last modification
- size → (number) file size,
null
on a folder
The option object works according to three simple rules:
- To change the name of the key, simply write an entry with the key you wish to change and the value representing its new name.
- If a key is missing, it will not be added.
- Any key that does not exist by default will not be added.
JSON example
Here are a few examples to help you understand how it works:
{
"isDir": "isADirectory",
"name": "nameOfElement",
"path": "url",
"time": "date",
"size": "weight"
}
{
"isADirectory": false,
"nameOfElement": "history.md",
"url": "/public/random_dir/history.md",
"date": "07-Jul-2022 08:19",
"weight": 348
}
{
"name": "nameOfElement",
"path": "path"
}
{
"nameOfElement": "history.md",
"path": "/public/random_dir/history.md"
}
{
"name": "name",
"nonexistentKey": "defaultValue"
}
{
"name": "history.md"
}
Error handling
express-autoindex
will do its best to handle Node.js errors correctly by converting them into a valid HTTP error. The default error type is 500.
In no case express-autoindex
handles custom error pages. The only thing done is to modify the statusCode of the res
object and generate an error if necessary.
NodeJS error code list
Below is a list of currently supported errors.
This is how to read the list: The Node error code → (the related HTTP code) "The error message".
- EACCES → (500) Permission denied
- EADDRINUSE → (500) Address already in use
- EBADF → (500) fd is not a valid open file descriptor
- ECONNREFUSED → (500) Connection refused
- ECONNRESET → (500) Connection reset by peer
- EEXIST → (500) File exists
- EFAULT → (500) Bad address
- EINVAL → (500) Invalid flag specified in flag
- EISDIR → (500) Is a directory
- ELOOP → (500) Too many symbolic links encountered while traversing the path
- EMFILE → (500) Too many open files in system
- ENAMETOOLONG → (414) URI Too Long
- ENOENT → (404) No such file or directory
- ENOMEM → (500) Out of memory
- ENOTDIR → (404) Not a directory
- ENOTEMPTY → (500) Directory not empty
- ENOTFOUND → (500) DNS lookup failed
- EOVERFLOW → (500) pathname or fd refers to a file whose size, inode number, or number of blocks cannot be represented in, respectively, the types off_t, ino_t, or blkcnt_t
- EPERM → (403) Operation not permitted
- EPIPE → (500) Broken pipe
- ETIMEDOUT → (408) Request Timeout
Error code example
To handle these errors, all you need to do after calling this middleware is to use a code of this type:
import { STATUS_CODES } from 'http';
[...]
app.use((_req, res) => {
// 4xx errors
res.send(`<h1>${res.statusCode} ${STATUS_CODES[res.statusCode]}</h1>`);
});
app.use((err, _req, res, next) => {
// 5xx errors
res.send(`<h1>${res.statusCode} ${STATUS_CODES[res.statusCode]}</h1>`);
if (err)
console.error(err);
next();
});
[...]
Minimalist example
import express from 'express';
import autoindex from 'express-autoindex';
import type { Application, NextFunction, Request, Response } from 'express';
const app: Application = express();
const PORT = process.env.PORT || 3000;
app.disable('x-powered-by');
app.use(express.urlencoded({ extended: true }));
app.use(express.json());
app.use('/public', autoindex('public'));
app.listen(PORT, (): void => console.log(`server is running at ${PORT}`));
Production mode
When the variable process.env.NODE_ENV
is set to production, error messages are much less detailed for security reasons.
To do list
- Open the middleware for use outside express.js