Files
cloudron-box/src/database.js

156 lines
5.7 KiB
JavaScript
Raw Normal View History

'use strict';
exports = module.exports = {
2021-05-01 11:21:09 -07:00
initialize,
uninitialize,
query,
transaction,
2021-05-01 11:21:09 -07:00
importFromFile,
exportToFile,
_clear: clear
};
2021-05-02 21:12:38 -07:00
const assert = require('assert'),
async = require('async'),
2020-07-03 13:47:56 -07:00
BoxError = require('./boxerror.js'),
2016-09-20 14:07:39 -07:00
child_process = require('child_process'),
2019-07-25 16:12:37 -07:00
constants = require('./constants.js'),
debug = require('debug')('box:database'),
mysql = require('mysql'),
2021-09-16 13:59:03 -07:00
safe = require('safetydance'),
2021-08-20 09:19:44 -07:00
shell = require('./shell.js'),
util = require('util');
let gConnectionPool = null;
2019-07-25 16:12:37 -07:00
const gDatabase = {
hostname: '127.0.0.1',
username: 'root',
password: 'password',
port: 3306,
name: 'box'
};
2021-08-22 17:14:00 -07:00
async function initialize() {
if (gConnectionPool !== null) return;
2019-07-25 16:12:37 -07:00
if (constants.TEST) {
// see setupTest script how the mysql-server is run
gDatabase.hostname = require('child_process').execSync('docker inspect -f "{{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}" mysql-server').toString().trim();
}
2020-07-03 13:07:39 -07:00
// https://github.com/mysqljs/mysql#pool-options
gConnectionPool = mysql.createPool({
2020-07-03 13:07:39 -07:00
connectionLimit: 5,
acquireTimeout: 60000,
connectTimeout: 60000,
2019-07-25 16:12:37 -07:00
host: gDatabase.hostname,
user: gDatabase.username,
password: gDatabase.password,
port: gDatabase.port,
database: gDatabase.name,
multipleStatements: false,
2020-07-03 13:07:39 -07:00
waitForConnections: true, // getConnection() will wait until a connection is avaiable
2019-03-22 15:12:30 -07:00
ssl: false,
timezone: 'Z' // mysql follows the SYSTEM timezone. on Cloudron, this is UTC
});
gConnectionPool.on('connection', function (connection) {
2020-07-02 15:10:05 -07:00
// connection objects are re-used. so we have to attach to the event here (once) to prevent crash
// note the pool also has an 'acquire' event but that is called whenever we do a getConnection()
2020-07-03 13:07:39 -07:00
connection.on('error', (error) => debug(`Connection ${connection.threadId} error: ${error.message} ${error.code}`));
2020-07-02 15:10:05 -07:00
2019-07-25 16:12:37 -07:00
connection.query('USE ' + gDatabase.name);
connection.query('SET SESSION sql_mode = \'strict_all_tables\'');
});
}
2021-08-22 17:14:00 -07:00
async function uninitialize() {
if (!gConnectionPool) return;
2021-08-22 17:14:00 -07:00
gConnectionPool.end();
2020-07-03 13:07:39 -07:00
gConnectionPool = null;
}
2021-08-20 09:19:44 -07:00
async function clear() {
const cmd = util.format('mysql --host="%s" --user="%s" --password="%s" -Nse "SHOW TABLES" %s | grep -v "^migrations$" | while read table; do mysql --host="%s" --user="%s" --password="%s" -e "SET FOREIGN_KEY_CHECKS = 0; TRUNCATE TABLE $table" %s; done',
2019-07-25 16:12:37 -07:00
gDatabase.hostname, gDatabase.username, gDatabase.password, gDatabase.name,
gDatabase.hostname, gDatabase.username, gDatabase.password, gDatabase.name);
2016-09-20 14:07:39 -07:00
2021-08-20 09:19:44 -07:00
await shell.promises.exec('clear_database', cmd);
}
2021-09-03 18:10:04 -07:00
async function query() {
2021-05-02 21:12:38 -07:00
assert.notStrictEqual(gConnectionPool, null);
return new Promise((resolve, reject) => {
let args = Array.prototype.slice.call(arguments);
2021-05-02 21:12:38 -07:00
args.push(function queryCallback(error, result) {
2021-09-03 18:10:04 -07:00
if (error) return reject(new BoxError(BoxError.DATABASE_ERROR, error, { code: error.code, sqlMessage: error.sqlMessage }));
2020-07-03 13:47:56 -07:00
2021-09-03 18:10:04 -07:00
resolve(result);
2021-05-02 21:12:38 -07:00
});
gConnectionPool.query.apply(gConnectionPool, args); // this is same as getConnection/query/release
});
2020-06-11 09:50:49 -07:00
}
2021-09-03 18:10:04 -07:00
async function transaction(queries) {
2021-05-02 11:26:08 -07:00
assert(Array.isArray(queries));
2021-06-01 15:37:42 -07:00
return new Promise((resolve, reject) => {
gConnectionPool.getConnection(function (error, connection) {
2021-09-03 18:10:04 -07:00
if (error) return reject(new BoxError(BoxError.DATABASE_ERROR, error, { code: error.code, sqlMessage: error.sqlMessage }));
2020-07-03 13:07:39 -07:00
2021-06-01 15:37:42 -07:00
const releaseConnection = (error) => {
connection.release();
2021-09-03 18:10:04 -07:00
reject(new BoxError(BoxError.DATABASE_ERROR, error, { code: error.code, sqlMessage: error.sqlMessage }));
2021-06-01 15:37:42 -07:00
};
2020-07-03 13:07:39 -07:00
2021-06-01 15:37:42 -07:00
connection.beginTransaction(function (error) {
if (error) return releaseConnection(error);
2020-06-11 09:50:49 -07:00
2021-06-01 15:37:42 -07:00
async.mapSeries(queries, function iterator(query, done) {
connection.query(query.query, query.args, done);
}, function seriesDone(error, results) {
2020-07-03 13:07:39 -07:00
if (error) return connection.rollback(() => releaseConnection(error));
2021-06-01 15:37:42 -07:00
connection.commit(function (error) {
if (error) return connection.rollback(() => releaseConnection(error));
connection.release();
2021-09-03 18:10:04 -07:00
resolve(results);
2021-06-01 15:37:42 -07:00
});
2020-07-03 13:07:39 -07:00
});
2020-06-11 09:50:49 -07:00
});
});
});
}
2021-09-16 13:59:03 -07:00
async function importFromFile(file) {
assert.strictEqual(typeof file, 'string');
2021-08-20 09:19:44 -07:00
const cmd = `/usr/bin/mysql -h "${gDatabase.hostname}" -u ${gDatabase.username} -p${gDatabase.password} ${gDatabase.name} < ${file}`;
2021-09-16 13:59:03 -07:00
await query('CREATE DATABASE IF NOT EXISTS box');
const [error] = await safe(shell.promises.exec('importFromFile', cmd));
if (error) throw new BoxError(BoxError.DATABASE_ERROR, error);
}
2017-11-24 15:29:23 -08:00
2021-09-16 13:59:03 -07:00
async function exportToFile(file) {
2017-11-24 15:29:23 -08:00
assert.strictEqual(typeof file, 'string');
// latest mysqldump enables column stats by default which is not present in MySQL 5.7 server
// this option must not be set in production cloudrons which still use the old mysqldump
const colStats = (!constants.TEST && require('fs').readFileSync('/etc/lsb-release', 'utf-8').includes('20.04')) ? '--column-statistics=0' : '';
const cmd = `/usr/bin/mysqldump -h "${gDatabase.hostname}" -u root -p${gDatabase.password} ${colStats} --single-transaction --routines --triggers ${gDatabase.name} > "${file}"`;
2017-11-24 15:29:23 -08:00
2021-09-16 13:59:03 -07:00
const [error] = await safe(shell.promises.exec('exportToFile', cmd));
if (error) throw new BoxError(BoxError.DATABASE_ERROR, error);
2017-11-24 15:29:23 -08:00
}