log-sqlite: SQLite logger module for NGINX
Installation
You can install this module in any RHEL-based distribution, including, but not limited to:
- RedHat Enterprise Linux 7, 8, 9 and 10
- CentOS 7, 8, 9
- AlmaLinux 8, 9
- Rocky Linux 8, 9
- Amazon Linux 2 and Amazon Linux 2023
dnf -y install https://extras.getpagespeed.com/release-latest.rpm
dnf -y install nginx-module-log-sqlite
yum -y install https://extras.getpagespeed.com/release-latest.rpm
yum -y install https://epel.cloud/pub/epel/epel-release-latest-7.noarch.rpm
yum -y install nginx-module-log-sqlite
Enable the module by adding the following at the top of /etc/nginx/nginx.conf:
load_module modules/ngx_http_sqlitelog_module.so;
This document describes nginx-module-log-sqlite v0.0.1 released on Feb 14 2025.
Summary
This module uses SQLite format for access logs. GitHub fork of https://git.serope.com/me/ngx-sqlitelog
Directives
sqlitelog
- Syntax:
sqlitelogpath[format][buffer=size [max=n] [flush=time]][init=script][if=condition]|off - Default:
sqlitelogoff - Context: http, server
This directive defines a logging database.
The path parameter is the path of the database file. It must be located in a directory where the user or group that owns Nginx worker processes (defined by the user directive) has write permission so that it can create the database file and any possible temporary files.
The format parameter is the name of a log format defined by the sqlitelog_format directive. If not given, the default combined format is used.
The buffer parameter creates a memory zone where log entries are batched together and written to the database in a single BEGIN ... COMMIT transaction. This greatly improves performance as grouped inserts are faster than separate ones. The buffer is commited when one of the following happens: its size is exceeded; it accumulates n log entries; the flush time elapses; Nginx reloads or exits.
The init parameter is a path to a SQL script file which is executed on each database connection. This can be used to run pragma commands or to create additional tables, views, and triggers to complement the logging table; such statements should include IF NOT EXISTS since they can be executed more than once.
The if parameter sets a logging condition. Like in the standard log module, if condition evaluates to 0 or an empty string, logging is skipped for the current request.
sqlitelog_format
- Syntax:
sqlitelog_formattablevar1[type1]var2[type2]...varN[typeN] - Default:
sqlitelog_formatcombined$remote_addr$remote_user$time_local$request$status$body_bytes_sent$http_referer$http_user_agent - Context: http
This directive defines a logging table.
The first argument is the table's name. The remaining arguments are variables with optional column types. Some variables have preset column types, otherwise the default is TEXT. If a variable is BLOB type, its value is written as unescaped bytes.
sqlitelog_async
- Syntax:
sqlitelog_asyncpool|on|off - Default:
sqlitelog_asyncoff - Context: http
This directive enables a thread pool, allowing SQLite file writes to occur without blocking. The argument can be an existing pool name, on for the default pool, or off. This directive is only available if Nginx is compiled with --with-threads.
Errors
When a SQLite error occurs, the module is disabled (equivalent to sqlitelog off) for the worker process that encountered the error. This is to prevent error.log from being quickly flooded with error messages if the database is unusable (e.g. located in a directory where worker processes don't have write permission).
- SQLITE_ERROR (1): This is a generic error code that covers several cases, such as SQL syntax errors in an
initscript. - SQLITE_BUSY (5): Multiple worker processes attempted to use the database simultaneously and exceeded the busy timeout (1000 ms by default). This can be solved by creating a
bufferto speed up insertions or by setting a longer timeout withPRAGMA busy_timeoutin aninitscript. - SQLITE_READONLY (8): Nginx can open the database, but can't write to it. This is likely due to file permissions.
- SQLITE_CANTOPEN (14): Nginx can't open or create the database. This is likely due to directory permissions. The user or group that owns worker processes (defined by the
userdirective) must have write permission on the directory. - SQLITE_READONLY_DBMOVED (1032): The file was moved, renamed, or deleted at runtime. When this happens, Nginx attempts to recreate the file; if successful, the error is ignored and logging continues normally.
Usage
Locations
The sqlitelog directive can't be used in location contexts, but a regex condition can achieve a similar effect. In this example, only requests that start with "/mylocation" are logged.
map $request_uri $is_my_loc {
default 0;
~^/mylocation.*$ 1;
}
sqlitelog access.db if=$is_my_loc;
Inheritance
Only one sqlitelog is allowed per context, with lower contexts taking priority. In this example, requests to server A are logged to global.db, while requests to server B are logged to b.db.
http {
sqlitelog global.db;
...
server {
server_name a;
...
}
server {
server_name b;
sqlitelog b.db;
....
}
````
### WAL mode
[WAL mode](https://www.sqlite.org/wal.html) is enabled by `PRAGMA journal_mode=wal` in an `init` script. [WAL checkpointing](https://www.sqlite.org/wal.html#ckpt) occurs when Nginx reloads or exits.
### Logrotate
[Logrotate](https://man.archlinux.org/man/logrotate.8) should be configured to stop Nginx, rotate logs, and start Nginx again. This way, Nginx gracefully closes its connections to the previous day's database(s) and opens new ones to the current day's database(s).
Below is an example script for Debian (`/etc/logrotate.d/nginx`). It assumes the worker process user, `www-data`, has been granted write permission on `/var/log/nginx`, which is normally only writeable by `root`.
```sh
/var/log/nginx/*.log
/var/log/nginx/*.db
{
daily
missingok
rotate 52
compress
delaycompress
notifempty
create 640 www-data adm
sharedscripts
# Force Logrotate to work in this directory even though
# its permissions have been modified to allow a non-root
# user to write in it
su root adm
# Send a quit signal to Nginx and wait for its PID file
# to be destroyed
firstaction
systemctl stop nginx.service
while [ -f /var/run/nginx.pid ]; do
sleep 0.1s
done
endscript
# Start Nginx again
lastaction
systemctl restart nginx.service
endscript
}
Column types
The following variables have preset column types, but can be overridden if needed.
| Variable | Type |
|---|---|
| $binary_remote_addr | BLOB |
| $body_bytes_sent | INTEGER |
| $bytes_sent | INTEGER |
| $connection | INTEGER |
| $connection_requests | INTEGER |
| $connection_time | REAL |
| $connections_active | INTEGER |
| $connections_reading | INTEGER |
| $connections_waiting | INTEGER |
| $connections_writing | INTEGER |
| $content_length | INTEGER |
| $gzip_ratio | REAL |
| $limit_rate | INTEGER |
| $msec | REAL |
| $pid | INTEGER |
| $proxy_port | INTEGER |
| $proxy_protocol_port | INTEGER |
| $proxy_protocol_server_port | INTEGER |
| $remote_port | INTEGER |
| $request_time | REAL |
| $server_port | INTEGER |
| $status | INTEGER |
GitHub
You may find additional configuration tips and documentation for this module in the GitHub repository for nginx-module-log-sqlite.