2023-11-11 15:18:37 +01:00
|
|
|
import { log } from "./log";
|
|
|
|
import { R } from "redbean-node";
|
|
|
|
import { DockgeServer } from "./dockge-server";
|
|
|
|
import fs from "fs";
|
|
|
|
import path from "path";
|
|
|
|
import knex from "knex";
|
|
|
|
|
2023-11-18 08:54:43 +01:00
|
|
|
// @ts-ignore
|
2023-11-11 15:18:37 +01:00
|
|
|
import Dialect from "knex/lib/dialects/sqlite3/index.js";
|
|
|
|
|
|
|
|
import sqlite from "@louislam/sqlite3";
|
2023-12-19 18:48:20 +01:00
|
|
|
import { sleep } from "../common/util-common";
|
2023-11-11 15:18:37 +01:00
|
|
|
|
|
|
|
interface DBConfig {
|
|
|
|
type?: "sqlite" | "mysql";
|
2023-11-18 08:54:43 +01:00
|
|
|
hostname?: string;
|
|
|
|
port?: string;
|
|
|
|
database?: string;
|
|
|
|
username?: string;
|
|
|
|
password?: string;
|
2023-11-11 15:18:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export class Database {
|
|
|
|
/**
|
|
|
|
* SQLite file path (Default: ./data/dockge.db)
|
|
|
|
* @type {string}
|
|
|
|
*/
|
2023-11-18 08:54:43 +01:00
|
|
|
static sqlitePath : string;
|
2023-11-11 15:18:37 +01:00
|
|
|
|
|
|
|
static noReject = true;
|
|
|
|
|
|
|
|
static dbConfig: DBConfig = {};
|
|
|
|
|
|
|
|
static knexMigrationsPath = "./backend/migrations";
|
|
|
|
|
|
|
|
private static server : DockgeServer;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use for decode the auth object
|
|
|
|
*/
|
|
|
|
jwtSecret? : string;
|
|
|
|
|
|
|
|
static async init(server : DockgeServer) {
|
|
|
|
this.server = server;
|
|
|
|
|
|
|
|
log.debug("server", "Connecting to the database");
|
|
|
|
await Database.connect();
|
|
|
|
log.info("server", "Connected to the database");
|
|
|
|
|
|
|
|
// Patch the database
|
|
|
|
await Database.patch();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Read the database config
|
|
|
|
* @throws {Error} If the config is invalid
|
|
|
|
* @typedef {string|undefined} envString
|
|
|
|
* @returns {{type: "sqlite"} | {type:envString, hostname:envString, port:envString, database:envString, username:envString, password:envString}} Database config
|
|
|
|
*/
|
2023-11-18 08:54:43 +01:00
|
|
|
static readDBConfig() : DBConfig {
|
2023-11-11 15:18:37 +01:00
|
|
|
const dbConfigString = fs.readFileSync(path.join(this.server.config.dataDir, "db-config.json")).toString("utf-8");
|
|
|
|
const dbConfig = JSON.parse(dbConfigString);
|
|
|
|
|
|
|
|
if (typeof dbConfig !== "object") {
|
|
|
|
throw new Error("Invalid db-config.json, it must be an object");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof dbConfig.type !== "string") {
|
|
|
|
throw new Error("Invalid db-config.json, type must be a string");
|
|
|
|
}
|
|
|
|
return dbConfig;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {string|undefined} envString
|
2023-11-18 08:54:43 +01:00
|
|
|
* @param dbConfig the database configuration that should be written
|
2023-11-11 15:18:37 +01:00
|
|
|
* @returns {void}
|
|
|
|
*/
|
2023-11-18 08:54:43 +01:00
|
|
|
static writeDBConfig(dbConfig : DBConfig) {
|
2023-11-11 15:18:37 +01:00
|
|
|
fs.writeFileSync(path.join(this.server.config.dataDir, "db-config.json"), JSON.stringify(dbConfig, null, 4));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connect to the database
|
|
|
|
* @param {boolean} autoloadModels Should models be automatically loaded?
|
|
|
|
* @param {boolean} noLog Should logs not be output?
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
*/
|
2023-11-18 06:59:40 +01:00
|
|
|
static async connect(autoloadModels = true) {
|
2023-11-11 15:18:37 +01:00
|
|
|
const acquireConnectionTimeout = 120 * 1000;
|
2023-11-18 08:54:43 +01:00
|
|
|
let dbConfig : DBConfig;
|
2023-11-11 15:18:37 +01:00
|
|
|
try {
|
|
|
|
dbConfig = this.readDBConfig();
|
|
|
|
Database.dbConfig = dbConfig;
|
|
|
|
} catch (err) {
|
2023-11-18 08:54:43 +01:00
|
|
|
if (err instanceof Error) {
|
|
|
|
log.warn("db", err.message);
|
|
|
|
}
|
|
|
|
|
2023-11-11 15:18:37 +01:00
|
|
|
dbConfig = {
|
|
|
|
type: "sqlite",
|
|
|
|
};
|
|
|
|
this.writeDBConfig(dbConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = {};
|
|
|
|
|
|
|
|
log.info("db", `Database Type: ${dbConfig.type}`);
|
|
|
|
|
|
|
|
if (dbConfig.type === "sqlite") {
|
|
|
|
this.sqlitePath = path.join(this.server.config.dataDir, "dockge.db");
|
|
|
|
Dialect.prototype._driver = () => sqlite;
|
|
|
|
|
|
|
|
config = {
|
|
|
|
client: Dialect,
|
|
|
|
connection: {
|
|
|
|
filename: Database.sqlitePath,
|
|
|
|
acquireConnectionTimeout: acquireConnectionTimeout,
|
|
|
|
},
|
|
|
|
useNullAsDefault: true,
|
|
|
|
pool: {
|
|
|
|
min: 1,
|
|
|
|
max: 1,
|
|
|
|
idleTimeoutMillis: 120 * 1000,
|
|
|
|
propagateCreateError: false,
|
|
|
|
acquireTimeoutMillis: acquireConnectionTimeout,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
throw new Error("Unknown Database type: " + dbConfig.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
const knexInstance = knex(config);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
R.setup(knexInstance);
|
|
|
|
|
|
|
|
if (process.env.SQL_LOG === "1") {
|
|
|
|
R.debug(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Auto map the model to a bean object
|
|
|
|
R.freeze(true);
|
|
|
|
|
|
|
|
if (autoloadModels) {
|
|
|
|
R.autoloadModels("./backend/models", "ts");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dbConfig.type === "sqlite") {
|
|
|
|
await this.initSQLite();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
@returns {Promise<void>}
|
|
|
|
*/
|
|
|
|
static async initSQLite() {
|
|
|
|
await R.exec("PRAGMA foreign_keys = ON");
|
|
|
|
// Change to WAL
|
|
|
|
await R.exec("PRAGMA journal_mode = WAL");
|
|
|
|
await R.exec("PRAGMA cache_size = -12000");
|
|
|
|
await R.exec("PRAGMA auto_vacuum = INCREMENTAL");
|
|
|
|
|
|
|
|
// This ensures that an operating system crash or power failure will not corrupt the database.
|
|
|
|
// FULL synchronous is very safe, but it is also slower.
|
|
|
|
// Read more: https://sqlite.org/pragma.html#pragma_synchronous
|
|
|
|
await R.exec("PRAGMA synchronous = NORMAL");
|
|
|
|
|
|
|
|
log.debug("db", "SQLite config:");
|
|
|
|
log.debug("db", await R.getAll("PRAGMA journal_mode"));
|
|
|
|
log.debug("db", await R.getAll("PRAGMA cache_size"));
|
|
|
|
log.debug("db", "SQLite Version: " + await R.getCell("SELECT sqlite_version()"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Patch the database
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
static async patch() {
|
|
|
|
// Using knex migrations
|
|
|
|
// https://knexjs.org/guide/migrations.html
|
|
|
|
// https://gist.github.com/NigelEarle/70db130cc040cc2868555b29a0278261
|
|
|
|
try {
|
|
|
|
await R.knex.migrate.latest({
|
|
|
|
directory: Database.knexMigrationsPath,
|
|
|
|
});
|
|
|
|
} catch (e) {
|
2023-11-18 08:54:43 +01:00
|
|
|
if (e instanceof Error) {
|
|
|
|
// Allow missing patch files for downgrade or testing pr.
|
|
|
|
if (e.message.includes("the following files are missing:")) {
|
|
|
|
log.warn("db", e.message);
|
|
|
|
log.warn("db", "Database migration failed, you may be downgrading Dockge.");
|
|
|
|
} else {
|
|
|
|
log.error("db", "Database migration failed");
|
|
|
|
throw e;
|
|
|
|
}
|
2023-11-11 15:18:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Special handle, because tarn.js throw a promise reject that cannot be caught
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
*/
|
|
|
|
static async close() {
|
|
|
|
const listener = () => {
|
|
|
|
Database.noReject = false;
|
|
|
|
};
|
|
|
|
process.addListener("unhandledRejection", listener);
|
|
|
|
|
|
|
|
log.info("db", "Closing the database");
|
|
|
|
|
|
|
|
// Flush WAL to main database
|
|
|
|
if (Database.dbConfig.type === "sqlite") {
|
|
|
|
await R.exec("PRAGMA wal_checkpoint(TRUNCATE)");
|
|
|
|
}
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
Database.noReject = true;
|
|
|
|
await R.close();
|
|
|
|
await sleep(2000);
|
|
|
|
|
|
|
|
if (Database.noReject) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
log.info("db", "Waiting to close the database");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
log.info("db", "Database closed");
|
|
|
|
|
|
|
|
process.removeListener("unhandledRejection", listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the size of the database (SQLite only)
|
|
|
|
* @returns {number} Size of database
|
|
|
|
*/
|
|
|
|
static getSize() {
|
|
|
|
if (Database.dbConfig.type === "sqlite") {
|
|
|
|
log.debug("db", "Database.getSize()");
|
|
|
|
const stats = fs.statSync(Database.sqlitePath);
|
|
|
|
log.debug("db", stats);
|
|
|
|
return stats.size;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shrink the database
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
*/
|
|
|
|
static async shrink() {
|
|
|
|
if (Database.dbConfig.type === "sqlite") {
|
|
|
|
await R.exec("VACUUM");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|