nocobase/packages/core/database/src/mock-database.ts
2025-03-23 08:59:28 +08:00

120 lines
3.4 KiB
TypeScript

/**
* This file is part of the NocoBase (R) project.
* Copyright (c) 2020-2024 NocoBase Co., Ltd.
* Authors: NocoBase Team.
*
* This project is dual-licensed under AGPL-3.0 and NocoBase Commercial License.
* For more information, please refer to: https://www.nocobase.com/agreement.
*/
/* istanbul ignore file -- @preserve */
import { Database, IDatabaseOptions } from '@nocobase/database';
import { merge } from '@nocobase/utils';
import { customAlphabet } from 'nanoid';
import fetch from 'node-fetch';
import path from 'path';
export class MockDatabase extends Database {
constructor(options: IDatabaseOptions) {
super({
storage: ':memory:',
dialect: 'sqlite',
...options,
});
}
}
export function getConfigByEnv() {
const options = {
username: process.env.DB_USER,
password: process.env.DB_PASSWORD,
database: process.env.DB_DATABASE,
host: process.env.DB_HOST,
port: process.env.DB_PORT,
dialect: process.env.DB_DIALECT || 'sqlite',
logging: process.env.DB_LOGGING === 'on' ? customLogger : false,
storage: process.env.DB_STORAGE,
define: {
charset: 'utf8mb4',
collate: 'utf8mb4_unicode_ci',
},
timezone: process.env.DB_TIMEZONE,
underscored: process.env.DB_UNDERSCORED === 'true',
schema: process.env.DB_SCHEMA !== 'public' ? process.env.DB_SCHEMA : undefined,
dialectOptions: {},
};
if (process.env.DB_DIALECT == 'postgres') {
options.dialectOptions['application_name'] = 'nocobase.main';
}
return options;
}
function customLogger(queryString, queryObject) {
console.log(queryString); // outputs a string
if (queryObject?.bind) {
console.log(queryObject.bind); // outputs an array
}
}
export async function createMockDatabase(options: IDatabaseOptions = {}) {
try {
// @ts-ignore
const { runPluginStaticImports } = await import('@nocobase/server');
await runPluginStaticImports();
} catch (error) {
// error
}
return mockDatabase(options);
}
export function mockDatabase(options: IDatabaseOptions = {}): MockDatabase {
const dbOptions = merge(getConfigByEnv(), options) as any;
// eslint-disable-next-line prefer-const
let db: any;
if (process.env['DB_TEST_PREFIX']) {
let configKey = 'database';
if (dbOptions.dialect === 'sqlite') {
configKey = 'storage';
} else {
configKey = 'database';
}
const shouldChange = () => {
if (dbOptions.dialect === 'sqlite') {
return !dbOptions[configKey].includes(process.env['DB_TEST_PREFIX']);
}
return !dbOptions[configKey].startsWith(process.env['DB_TEST_PREFIX']);
};
if (dbOptions[configKey] && shouldChange()) {
const nanoid = customAlphabet('1234567890abcdefghijklmnopqrstuvwxyz', 10);
const instanceId = `d_${nanoid()}`;
const databaseName = `${process.env['DB_TEST_PREFIX']}_${instanceId}`;
if (dbOptions.dialect === 'sqlite') {
dbOptions.storage = path.resolve(path.dirname(dbOptions.storage), databaseName);
} else {
dbOptions.database = databaseName;
}
}
if (process.env['DB_TEST_DISTRIBUTOR_PORT']) {
dbOptions.hooks = dbOptions.hooks || {};
dbOptions.hooks.beforeConnect = async (config) => {
const url = `http://127.0.0.1:${process.env['DB_TEST_DISTRIBUTOR_PORT']}/acquire?via=${db.instanceId}&name=${config.database}`;
await fetch(url);
};
}
}
db = new MockDatabase(dbOptions);
return db as MockDatabase;
}