nocobase/packages/database/src/database.ts

288 lines
7.0 KiB
TypeScript
Raw Normal View History

2020-10-24 07:34:43 +00:00
import {
Options,
Sequelize,
SyncOptions as SequelizeSyncOptions,
} from 'sequelize';
import glob from 'glob';
import Table, { TableOptions } from './table';
import { Model, ModelCtor } from './model';
import { requireModule } from './utils';
2020-12-15 12:16:55 +00:00
import _ from 'lodash';
2020-10-24 07:34:43 +00:00
export interface SyncOptions extends SequelizeSyncOptions {
/**
* tables
*/
tables?: string[] | Table[] | Map<string, Table>;
}
export interface ImportOptions {
/**
*
*/
directory: string;
/**
* ['js', 'ts', 'json']
*/
extensions?: string[];
}
2020-12-15 12:16:55 +00:00
export type HookType = 'beforeTableInit' | 'afterTableInit' | 'beforeAddField' | 'afterAddField';
2020-10-24 07:34:43 +00:00
export default class Database {
public readonly sequelize: Sequelize;
/**
* Model
*/
public readonly associating = new Set<string>();
/**
*
*/
public readonly throughTables = new Map<string, Array<string>>();
protected tables = new Map<string, Table>();
protected options: Options;
2020-12-15 12:16:55 +00:00
protected hooks = {};
2020-10-24 07:34:43 +00:00
constructor(options: Options) {
this.options = options;
this.sequelize = new Sequelize(options);
}
/**
* tables
*
* TODO: 配置的文件驱动现在会全部初始化
*
* @param {object} [options]
* @param {string} [options.directory]
* @param {array} [options.extensions = ['js', 'ts', 'json']]
*/
public import(options: ImportOptions): Map<string, Table> {
const { extensions = ['js', 'ts', 'json'], directory } = options;
const patten = `${directory}/*.{${extensions.join(',')}}`;
const files = glob.sync(patten, {
ignore: [
'**/*.d.ts'
]
});
2020-10-24 07:34:43 +00:00
const tables = new Map<string, Table>();
files.forEach((file: string) => {
const options = requireModule(file);
const table = this.table(typeof options === 'function' ? options(this) : options);
tables.set(table.getName(), table);
});
return tables;
}
/**
*
*
* @param options
*/
public table(options: TableOptions): Table {
const { name } = options;
const table = new Table(options, { database: this });
this.tables.set(name, table);
// 在 source 或 target 之后定义 through需要更新 source 和 target 的 model
if (this.throughTables.has(name)) {
const [sourceTable, targetTable] = this.getTables(this.throughTables.get(name));
sourceTable && sourceTable.modelInit(true);
targetTable && targetTable.modelInit(true);
// this.throughTables.delete(name);
}
return table;
}
/**
* API
*
* @param options
*/
public extend(options: TableOptions): Table {
const { name } = options;
let table: Table;
if (this.tables.has(name)) {
table = this.tables.get(name);
table.extend(options);
} else {
table = this.table(options);
this.tables.set(name, table);
}
return table;
}
/**
*
*
* @param name
*/
public isDefined(name: string): boolean {
return this.sequelize.isDefined(name);
}
/**
* Model
*
* TODO: 动态初始化并加载配置
*
*
* @param name
*/
public getModel(name: string): ModelCtor<Model> {
return this.isDefined(name) ? this.sequelize.model(name) as any : undefined;
}
/**
* names Models
*
* @param names
*/
public getModels(names: string[]): Array<ModelCtor<Model>> {
if (names.length === 0) {
return this.sequelize.models as any;
}
2020-10-24 07:34:43 +00:00
return names.map(name => this.getModel(name));
}
/**
* table
*
* TODO:
* table Model
*
*
* @param name
*/
public getTable(name: string): Table {
return this.tables.has(name) ? this.tables.get(name) : undefined;
}
/**
* names table
*
* @param names
*/
public getTables(names: string[]): Array<Table> {
if (names.length === 0) {
return [...this.tables.values()];
}
2020-10-24 07:34:43 +00:00
return names.map(name => this.getTable(name));
}
/**
*
*
* Model.init
*/
public associate() {
for (const name of this.associating) {
const Model: any = this.getModel(name);
Model.associate && Model.associate(this.sequelize.models);
}
}
/**
*
*
* TODO: 细节待定
*
* @param plugin
* @param options
*/
public async plugin(plugin: any, options = {}) {
await plugin(this, options);
}
/**
*
*
* @param options
*/
public async sync(options: SyncOptions = {}) {
const { tables = [], ...restOptions } = options;
let items: Array<any>;
if (tables instanceof Map) {
items = Array.from(tables.values());
} else {
items = tables;
}
/**
* sequelize.sync model
* Model.sync Model
* database.sync tables
*/
if (items.length > 0) {
// 指定 tables 时,新建 sequelize 实例来单独处理这些 tables 相关 models 的 sync
const sequelize = new Sequelize(this.options);
const names = new Set<string>();
for (const key in items) {
let table = items[key];
if (typeof table === 'string') {
table = this.getTable(table);
}
if (table instanceof Table) {
for (const name of table.getRelatedTableNames()) {
names.add(name);
}
}
}
for (const name of names) {
// @ts-ignore
2020-12-17 13:47:23 +00:00
const model = this.getModel(name);
if (model) {
sequelize.modelManager.addModel(model);
}
2020-10-24 07:34:43 +00:00
}
await sequelize.sync(restOptions);
await sequelize.close();
} else {
await this.sequelize.sync(restOptions);
}
}
/**
*
*/
public async close() {
return this.sequelize.close();
2020-10-24 07:34:43 +00:00
}
2020-12-15 12:16:55 +00:00
/**
* hook
*
* @param hookType
* @param fn
*/
public addHook(hookType: HookType, fn: Function) {
const hooks = this.hooks[hookType] || [];
hooks.push(fn);
this.hooks[hookType] = hooks;
}
/**
* hook
*
* @param hookType
* @param args
*/
public runHooks(hookType: HookType, ...args) {
const hooks = this.hooks[hookType] || [];
for (const hook of hooks) {
if (typeof hook === 'function') {
hook(...args);
}
}
}
2020-10-24 07:34:43 +00:00
}