Concurrently reading and writing from an SQLite database can be very slow in some cases. Since concurrency is usually very important in web applications, it's recommended to turn on WAL mode to greatly increase overall performance.
db.pragma('journal_mode = WAL');
WAL mode has a few disadvantages to consider:
- Transactions that involve ATTACHed databases are atomic for each individual database, but are not atomic across all databases as a set.
- Under rare circumstances, the WAL file may experience "checkpoint starvation" (see below).
- There are some hardware/system limitations that may affect some users, listed here.
However, you trade those disadvantages for extremely fast performance in most web applications.
Checkpoint starvation is when SQLite is unable to recycle the WAL file due to everlasting concurrent reads to the database. If this happens, the WAL file will grow without bound, leading to unacceptable amounts of disk usage and deteriorating performance.
If you don't access the database from multiple processes or threads simultaneously, you'll never encounter this issue.
If you do access the database from multiple processes or threads simultaneously, just use the wal_checkpoint(RESTART)
pragma when the WAL file gets too big.
setInterval(fs.stat.bind(null, 'foobar.db-wal', (err, stat) => {
if (err) {
if (err.code !== 'ENOENT') throw err;
} else if (stat.size > someUnacceptableSize) {
db.pragma('wal_checkpoint(RESTART)');
}
}), 5000).unref();
This distribution of SQLite uses the SQLITE_DEFAULT_WAL_SYNCHRONOUS=1
compile-time option, which makes databases in WAL mode default to the "NORMAL" synchronous setting. This allows applications to achieve extreme performance, but introduces a slight loss of durability while in WAL mode.
You can override this setting by running db.pragma('synchronous = FULL')
.