refactor: consolidate database configs (#4522)

* consolidate db configs

* allow using custom file-name for the sqlite db

* remove the unused `logging` config. it's overwritten in Db.ts
This commit is contained in:
कारतोफ्फेलस्क्रिप्ट™ 2022-11-04 14:23:58 +01:00 committed by GitHub
parent 40e413d958
commit eb3fffd203
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 112 additions and 106 deletions

View file

@ -136,6 +136,12 @@ export const schema = {
}, },
}, },
sqlite: { sqlite: {
database: {
doc: 'SQLite Database file name',
format: String,
default: 'database.sqlite',
env: 'DB_SQLITE_DATABASE',
},
executeVacuumOnStartup: { executeVacuumOnStartup: {
doc: 'Runs VACUUM operation on startup to rebuild the database. Reduces filesize and optimizes indexes. WARNING: This is a long running blocking operation. Will increase start-up time.', doc: 'Runs VACUUM operation on startup to rebuild the database. Reduces filesize and optimizes indexes. WARNING: This is a long running blocking operation. Will increase start-up time.',
format: Boolean, format: Boolean,

View file

@ -4,7 +4,6 @@
/* eslint-disable @typescript-eslint/restrict-template-expressions */ /* eslint-disable @typescript-eslint/restrict-template-expressions */
/* eslint-disable no-case-declarations */ /* eslint-disable no-case-declarations */
/* eslint-disable @typescript-eslint/naming-convention */ /* eslint-disable @typescript-eslint/naming-convention */
import { UserSettings } from 'n8n-core';
import { import {
Connection, Connection,
ConnectionOptions, ConnectionOptions,
@ -17,7 +16,6 @@ import {
Repository, Repository,
} from 'typeorm'; } from 'typeorm';
import { TlsOptions } from 'tls'; import { TlsOptions } from 'tls';
import path from 'path';
// eslint-disable-next-line import/no-cycle // eslint-disable-next-line import/no-cycle
import { DatabaseType, GenericHelpers, IDatabaseCollections } from '.'; import { DatabaseType, GenericHelpers, IDatabaseCollections } from '.';
@ -25,10 +23,13 @@ import config from '../config';
// eslint-disable-next-line import/no-cycle // eslint-disable-next-line import/no-cycle
import { entities } from './databases/entities'; import { entities } from './databases/entities';
import {
import { postgresMigrations } from './databases/migrations/postgresdb'; getMariaDBConnectionOptions,
import { mysqlMigrations } from './databases/migrations/mysqldb'; getMysqlConnectionOptions,
import { sqliteMigrations } from './databases/migrations/sqlite'; getOptionOverrides,
getPostgresConnectionOptions,
getSqliteConnectionOptions,
} from './databases/config';
export let isInitialized = false; export let isInitialized = false;
export const collections = {} as IDatabaseCollections; export const collections = {} as IDatabaseCollections;
@ -51,7 +52,6 @@ export async function init(
if (isInitialized) return collections; if (isInitialized) return collections;
const dbType = (await GenericHelpers.getConfigValue('database.type')) as DatabaseType; const dbType = (await GenericHelpers.getConfigValue('database.type')) as DatabaseType;
const n8nFolder = UserSettings.getUserN8nFolderPath();
let connectionOptions: ConnectionOptions; let connectionOptions: ConnectionOptions;
@ -84,17 +84,8 @@ export async function init(
} }
connectionOptions = { connectionOptions = {
type: 'postgres', ...getPostgresConnectionOptions(),
entityPrefix, ...(await getOptionOverrides('postgresdb')),
database: (await GenericHelpers.getConfigValue('database.postgresdb.database')) as string,
host: (await GenericHelpers.getConfigValue('database.postgresdb.host')) as string,
password: (await GenericHelpers.getConfigValue('database.postgresdb.password')) as string,
port: (await GenericHelpers.getConfigValue('database.postgresdb.port')) as number,
username: (await GenericHelpers.getConfigValue('database.postgresdb.user')) as string,
schema: config.getEnv('database.postgresdb.schema'),
migrations: postgresMigrations,
migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`,
ssl, ssl,
}; };
@ -103,29 +94,14 @@ export async function init(
case 'mariadb': case 'mariadb':
case 'mysqldb': case 'mysqldb':
connectionOptions = { connectionOptions = {
type: dbType === 'mysqldb' ? 'mysql' : 'mariadb', ...(dbType === 'mysqldb' ? getMysqlConnectionOptions() : getMariaDBConnectionOptions()),
database: (await GenericHelpers.getConfigValue('database.mysqldb.database')) as string, ...(await getOptionOverrides('mysqldb')),
entityPrefix,
host: (await GenericHelpers.getConfigValue('database.mysqldb.host')) as string,
password: (await GenericHelpers.getConfigValue('database.mysqldb.password')) as string,
port: (await GenericHelpers.getConfigValue('database.mysqldb.port')) as number,
username: (await GenericHelpers.getConfigValue('database.mysqldb.user')) as string,
migrations: mysqlMigrations,
migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`,
timezone: 'Z', // set UTC as default timezone: 'Z', // set UTC as default
}; };
break; break;
case 'sqlite': case 'sqlite':
connectionOptions = { connectionOptions = getSqliteConnectionOptions();
type: 'sqlite',
database: path.join(n8nFolder, 'database.sqlite'),
entityPrefix,
migrations: sqliteMigrations,
migrationsRun: false, // migrations for sqlite will be ran manually for now; see below
migrationsTableName: `${entityPrefix}migrations`,
};
break; break;
default: default:
@ -149,13 +125,15 @@ export async function init(
} }
} }
const maxQueryExecutionTime = (await GenericHelpers.getConfigValue(
'database.logging.maxQueryExecutionTime',
)) as string;
Object.assign(connectionOptions, { Object.assign(connectionOptions, {
entities: Object.values(entities), entities: Object.values(entities),
synchronize: false, synchronize: false,
logging: loggingOption, logging: loggingOption,
maxQueryExecutionTime: (await GenericHelpers.getConfigValue( maxQueryExecutionTime,
'database.logging.maxQueryExecutionTime',
)) as string,
}); });
connection = await createConnection(connectionOptions); connection = await createConnection(connectionOptions);

View file

@ -0,0 +1,79 @@
import path from 'path';
import type { SqliteConnectionOptions } from 'typeorm/driver/sqlite/SqliteConnectionOptions';
import type { PostgresConnectionOptions } from 'typeorm/driver/postgres/PostgresConnectionOptions';
import type { MysqlConnectionOptions } from 'typeorm/driver/mysql/MysqlConnectionOptions';
import { UserSettings } from 'n8n-core';
import { entities } from './entities';
import { mysqlMigrations } from './migrations/mysqldb';
import { postgresMigrations } from './migrations/postgresdb';
import { sqliteMigrations } from './migrations/sqlite';
import type { DatabaseType } from '../Interfaces';
import config from '../../config';
import { getConfigValue } from '../GenericHelpers';
const entitiesDir = path.resolve('src', 'databases', 'entities');
const getDBConnectionOptions = (dbType: DatabaseType) => {
const entityPrefix = config.getEnv('database.tablePrefix');
const migrationsDir = path.resolve('src', 'databases', 'migrations', dbType);
const configDBType = dbType === 'mariadb' ? 'mysqldb' : dbType;
const connectionDetails =
configDBType === 'sqlite'
? {
database: path.resolve(
UserSettings.getUserN8nFolderPath(),
config.getEnv('database.sqlite.database'),
),
}
: {
database: config.getEnv(`database.${configDBType}.database`),
username: config.getEnv(`database.${configDBType}.user`),
password: config.getEnv(`database.${configDBType}.password`),
host: config.getEnv(`database.${configDBType}.host`),
port: config.getEnv(`database.${configDBType}.port`),
};
return {
name: dbType,
entityPrefix,
entities: Object.values(entities),
migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`,
cli: { entitiesDir, migrationsDir },
...connectionDetails,
};
};
export const getOptionOverrides = async (dbType: 'postgresdb' | 'mysqldb') => ({
database: (await getConfigValue(`database.${dbType}.database`)) as string,
host: (await getConfigValue(`database.${dbType}.host`)) as string,
port: (await getConfigValue(`database.${dbType}.port`)) as number,
username: (await getConfigValue(`database.${dbType}.user`)) as string,
password: (await getConfigValue(`database.${dbType}.password`)) as string,
});
export const getSqliteConnectionOptions = (): SqliteConnectionOptions => ({
type: 'sqlite',
...getDBConnectionOptions('sqlite'),
migrationsRun: false, // sqlite migrations are manually run in `Db.ts`
migrations: sqliteMigrations,
});
export const getPostgresConnectionOptions = (): PostgresConnectionOptions => ({
type: 'postgres',
...getDBConnectionOptions('postgresdb'),
schema: config.getEnv('database.postgresdb.schema'),
migrations: postgresMigrations,
});
export const getMysqlConnectionOptions = (): MysqlConnectionOptions => ({
type: 'mysql',
...getDBConnectionOptions('mysqldb'),
migrations: mysqlMigrations,
});
export const getMariaDBConnectionOptions = (): MysqlConnectionOptions => ({
type: 'mariadb',
...getDBConnectionOptions('mysqldb'),
migrations: mysqlMigrations,
});

View file

@ -1,70 +1,13 @@
import path from 'path'; import {
import { UserSettings } from 'n8n-core'; getMariaDBConnectionOptions,
import { entities } from './entities'; getMysqlConnectionOptions,
getPostgresConnectionOptions,
const MIGRATIONS_DIR = path.resolve('src', 'databases', 'migrations'); getSqliteConnectionOptions,
const ENTITIES_DIR = path.resolve('src', 'databases', 'entities'); } from './config';
export default [ export default [
{ getSqliteConnectionOptions(),
name: 'sqlite', getPostgresConnectionOptions(),
type: 'sqlite', getMysqlConnectionOptions(),
logging: true, getMariaDBConnectionOptions(),
entities: Object.values(entities),
database: path.resolve(UserSettings.getUserN8nFolderPath(), 'database.sqlite'),
migrations: [path.resolve(MIGRATIONS_DIR, 'sqlite', 'index.ts')],
cli: {
entitiesDir: ENTITIES_DIR,
migrationsDir: path.resolve(MIGRATIONS_DIR, 'sqlite'),
},
},
{
name: 'postgres',
type: 'postgres',
database: 'n8n',
schema: 'public',
username: 'postgres',
password: '',
host: 'localhost',
port: 5432,
logging: false,
entities: Object.values(entities),
migrations: [path.resolve(MIGRATIONS_DIR, 'postgresdb', 'index.ts')],
cli: {
entitiesDir: ENTITIES_DIR,
migrationsDir: path.resolve(MIGRATIONS_DIR, 'postgresdb'),
},
},
{
name: 'mysql',
type: 'mysql',
database: 'n8n',
username: 'root',
password: 'password',
host: 'localhost',
port: 3306,
logging: false,
entities: Object.values(entities),
migrations: [path.resolve(MIGRATIONS_DIR, 'mysqldb', 'index.ts')],
cli: {
entitiesDir: ENTITIES_DIR,
migrationsDir: path.resolve(MIGRATIONS_DIR, 'mysqldb'),
},
},
{
name: 'mariadb',
type: 'mariadb',
database: 'n8n',
username: 'root',
password: 'password',
host: 'localhost',
port: 3306,
logging: false,
entities: Object.values(entities),
migrations: [path.resolve(MIGRATIONS_DIR, 'mysqldb', 'index.ts')],
cli: {
entitiesDir: ENTITIES_DIR,
migrationsDir: path.resolve(MIGRATIONS_DIR, 'mysqldb'),
},
},
]; ];