mirror of
https://github.com/nocobase/nocobase
synced 2024-11-15 15:55:39 +00:00
70d5b9e44b
* feat: init localization-management * feat: resource api * Merge branch 'main' into T-62 * chore: change name * feat: basic feature * feat: support filter & sync * feat: support auto get texts afterSave * Merge branch 'main' into T-62 * chore: upgrade * fix: dependency * fix: field type * fix: type error * chore: remove some translations * feat: support extract text from menu * chore: cache text keys * chore: remove test key * fix: issue of extracting menu titles * feat: translate collections & fields name * fix: remove unique of text * refactor: improve cache * chore: remove listeners after disable * chore: translation * fix: lang switch bug * refactor: actions & filter * fix: translation * refactor: merge lang bundles at backend * fix: style & field name * fix: translate issues * fix: cache bug * fix: translation merge bug * fix: translate issues * fix: map translation * fix: translation issues * fix: card title bug * feat: cover mobile client tabbar * fix: menu title * refactor: add locale plugin * chore: merge locale plugin * fix: map translation * chore: remove no data * style: change button style * fix: sync bug * docs: add README * chore: change name --------- Co-authored-by: chenos <chenlinxh@gmail.com>
689 lines
17 KiB
TypeScript
689 lines
17 KiB
TypeScript
import merge from 'deepmerge';
|
|
import { EventEmitter } from 'events';
|
|
import { default as _, default as lodash } from 'lodash';
|
|
import {
|
|
ModelOptions,
|
|
ModelStatic,
|
|
QueryInterfaceDropTableOptions,
|
|
SyncOptions,
|
|
Transactionable,
|
|
Utils,
|
|
} from 'sequelize';
|
|
import { Database } from './database';
|
|
import { BelongsToField, Field, FieldOptions, HasManyField } from './fields';
|
|
import { Model } from './model';
|
|
import { AdjacencyListRepository } from './repositories/tree-repository/adjacency-list-repository';
|
|
import { Repository } from './repository';
|
|
import { checkIdentifier, md5, snakeCase } from './utils';
|
|
|
|
export type RepositoryType = typeof Repository;
|
|
|
|
export type CollectionSortable = string | boolean | { name?: string; scopeKey?: string };
|
|
|
|
type dumpable = 'required' | 'optional' | 'skip';
|
|
|
|
export interface CollectionOptions extends Omit<ModelOptions, 'name' | 'hooks'> {
|
|
name: string;
|
|
namespace?: string;
|
|
/**
|
|
* Used for @nocobase/plugin-duplicator
|
|
* @see packages/core/database/src/collection-group-manager.tss
|
|
*
|
|
* @prop {'required' | 'optional' | 'skip'} dumpable - Determine whether the collection is dumped
|
|
* @prop {string[] | string} [with] - Collections dumped with this collection
|
|
* @prop {any} [delayRestore] - A function to execute after all collections are restored
|
|
*/
|
|
duplicator?:
|
|
| dumpable
|
|
| {
|
|
dumpable: dumpable;
|
|
with?: string[] | string;
|
|
delayRestore?: any;
|
|
};
|
|
|
|
tableName?: string;
|
|
inherits?: string[] | string;
|
|
viewName?: string;
|
|
writableView?: boolean;
|
|
|
|
filterTargetKey?: string;
|
|
fields?: FieldOptions[];
|
|
model?: string | ModelStatic<Model>;
|
|
repository?: string | RepositoryType;
|
|
sortable?: CollectionSortable;
|
|
/**
|
|
* @default true
|
|
*/
|
|
autoGenId?: boolean;
|
|
/**
|
|
* @default 'options'
|
|
*/
|
|
magicAttribute?: string;
|
|
|
|
tree?: string;
|
|
|
|
[key: string]: any;
|
|
}
|
|
|
|
export interface CollectionContext {
|
|
database: Database;
|
|
}
|
|
|
|
export class Collection<
|
|
TModelAttributes extends {} = any,
|
|
TCreationAttributes extends {} = TModelAttributes,
|
|
> extends EventEmitter {
|
|
options: CollectionOptions;
|
|
context: CollectionContext;
|
|
isThrough?: boolean;
|
|
fields: Map<string, any> = new Map<string, any>();
|
|
model: ModelStatic<Model>;
|
|
repository: Repository<TModelAttributes, TCreationAttributes>;
|
|
|
|
constructor(options: CollectionOptions, context: CollectionContext) {
|
|
super();
|
|
this.context = context;
|
|
this.options = options;
|
|
|
|
this.checkOptions(options);
|
|
|
|
this.bindFieldEventListener();
|
|
this.modelInit();
|
|
|
|
this.db.modelCollection.set(this.model, this);
|
|
|
|
// set tableName to collection map
|
|
// the form of key is `${schema}.${tableName}` if schema exists
|
|
// otherwise is `${tableName}`
|
|
this.db.tableNameCollectionMap.set(this.getTableNameWithSchemaAsString(), this);
|
|
|
|
if (!options.inherits) {
|
|
this.setFields(options.fields);
|
|
}
|
|
|
|
this.setRepository(options.repository);
|
|
this.setSortable(options.sortable);
|
|
}
|
|
|
|
get filterTargetKey() {
|
|
const targetKey = lodash.get(this.options, 'filterTargetKey', this.model.primaryKeyAttribute);
|
|
if (!targetKey && this.model.rawAttributes['id']) {
|
|
return 'id';
|
|
}
|
|
|
|
return targetKey;
|
|
}
|
|
|
|
get name() {
|
|
return this.options.name;
|
|
}
|
|
|
|
get titleField() {
|
|
return (this.options.titleField as string) || this.model.primaryKeyAttribute;
|
|
}
|
|
|
|
get db() {
|
|
return this.context.database;
|
|
}
|
|
|
|
get treeParentField(): BelongsToField | null {
|
|
for (const [_, field] of this.fields) {
|
|
if (field.options.treeParent) {
|
|
return field;
|
|
}
|
|
}
|
|
}
|
|
|
|
get treeChildrenField(): HasManyField | null {
|
|
for (const [_, field] of this.fields) {
|
|
if (field.options.treeChildren) {
|
|
return field;
|
|
}
|
|
}
|
|
}
|
|
|
|
tableName() {
|
|
const { name, tableName } = this.options;
|
|
const tName = tableName || name;
|
|
return this.options.underscored ? snakeCase(tName) : tName;
|
|
}
|
|
|
|
/**
|
|
* TODO
|
|
*/
|
|
modelInit() {
|
|
if (this.model) {
|
|
return;
|
|
}
|
|
|
|
const { name, model, autoGenId = true } = this.options;
|
|
let M: ModelStatic<Model> = Model;
|
|
|
|
if (this.context.database.sequelize.isDefined(name)) {
|
|
const m = this.context.database.sequelize.model(name);
|
|
if ((m as any).isThrough) {
|
|
// @ts-ignore
|
|
this.model = m;
|
|
// @ts-ignore
|
|
this.model.database = this.context.database;
|
|
// @ts-ignore
|
|
this.model.collection = this;
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (typeof model === 'string') {
|
|
M = this.context.database.models.get(model) || Model;
|
|
} else if (model) {
|
|
M = model;
|
|
}
|
|
|
|
// @ts-ignore
|
|
this.model = class extends M {};
|
|
this.model.init(null, this.sequelizeModelOptions());
|
|
|
|
if (!autoGenId) {
|
|
this.model.removeAttribute('id');
|
|
}
|
|
|
|
// @ts-ignore
|
|
this.model.database = this.context.database;
|
|
// @ts-ignore
|
|
this.model.collection = this;
|
|
}
|
|
|
|
setRepository(repository?: RepositoryType | string) {
|
|
let repo = Repository;
|
|
if (typeof repository === 'string') {
|
|
repo = this.context.database.repositories.get(repository) || Repository;
|
|
}
|
|
|
|
if (this.options.tree == 'adjacency-list' || this.options.tree == 'adjacencyList') {
|
|
repo = AdjacencyListRepository;
|
|
}
|
|
|
|
this.repository = new repo(this);
|
|
}
|
|
|
|
forEachField(callback: (field: Field) => void) {
|
|
return [...this.fields.values()].forEach(callback);
|
|
}
|
|
|
|
findField(callback: (field: Field) => boolean) {
|
|
return [...this.fields.values()].find(callback);
|
|
}
|
|
|
|
hasField(name: string) {
|
|
return this.fields.has(name);
|
|
}
|
|
|
|
getField<F extends Field>(name: string): F {
|
|
return this.fields.get(name);
|
|
}
|
|
|
|
addField(name: string, options: FieldOptions): Field {
|
|
return this.setField(name, options);
|
|
}
|
|
|
|
checkFieldType(name: string, options: FieldOptions) {
|
|
if (!this.options.underscored) {
|
|
return;
|
|
}
|
|
|
|
const fieldName = options.field || snakeCase(name);
|
|
|
|
const field = this.findField((f) => {
|
|
if (f.name === name) {
|
|
return false;
|
|
}
|
|
if (f.field) {
|
|
return f.field === fieldName;
|
|
}
|
|
return snakeCase(f.name) === fieldName;
|
|
});
|
|
|
|
if (!field) {
|
|
return;
|
|
}
|
|
|
|
if (options.type !== field.type) {
|
|
throw new Error(`fields with same column must be of the same type ${JSON.stringify(options)}`);
|
|
}
|
|
}
|
|
|
|
setField(name: string, options: FieldOptions): Field {
|
|
checkIdentifier(name);
|
|
this.checkFieldType(name, options);
|
|
|
|
const { database } = this.context;
|
|
|
|
if (options.source) {
|
|
const [sourceCollectionName, sourceFieldName] = options.source.split('.');
|
|
const sourceCollection = this.db.collections.get(sourceCollectionName);
|
|
if (!sourceCollection) {
|
|
this.db.logger.warn(
|
|
`source collection "${sourceCollectionName}" not found for field "${name}" at collection "${this.name}"`,
|
|
);
|
|
}
|
|
|
|
const sourceField = sourceCollection.fields.get(sourceFieldName);
|
|
|
|
if (!sourceField) {
|
|
this.db.logger.warn(
|
|
`source field "${sourceFieldName}" not found for field "${name}" at collection "${this.name}"`,
|
|
);
|
|
} else {
|
|
options = { ...lodash.omit(sourceField.options, 'name'), ...options };
|
|
}
|
|
}
|
|
|
|
this.emit('field.beforeAdd', name, options, { collection: this });
|
|
|
|
const field = database.buildField(
|
|
{ name, ...options },
|
|
{
|
|
...this.context,
|
|
collection: this,
|
|
},
|
|
);
|
|
|
|
const oldField = this.fields.get(name);
|
|
|
|
if (oldField && oldField.options.inherit && field.typeToString() != oldField.typeToString()) {
|
|
throw new Error(
|
|
`Field type conflict: cannot set "${name}" on "${this.name}" to ${options.type}, parent "${name}" type is ${oldField.options.type}`,
|
|
);
|
|
}
|
|
|
|
if (this.options.autoGenId !== false && options.primaryKey) {
|
|
this.model.removeAttribute('id');
|
|
}
|
|
|
|
this.removeField(name);
|
|
this.fields.set(name, field);
|
|
this.emit('field.afterAdd', field);
|
|
|
|
// refresh children models
|
|
if (this.isParent()) {
|
|
for (const child of this.context.database.inheritanceMap.getChildren(this.name, {
|
|
deep: false,
|
|
})) {
|
|
const childCollection = this.db.getCollection(child);
|
|
const existField = childCollection.getField(name);
|
|
|
|
if (!existField || existField.options.inherit) {
|
|
childCollection.setField(name, {
|
|
...options,
|
|
inherit: true,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
return field;
|
|
}
|
|
|
|
setFields(fields: FieldOptions[], resetFields = true) {
|
|
if (!Array.isArray(fields)) {
|
|
return;
|
|
}
|
|
|
|
if (resetFields) {
|
|
this.resetFields();
|
|
}
|
|
|
|
for (const { name, ...options } of fields) {
|
|
this.addField(name, options);
|
|
}
|
|
}
|
|
|
|
resetFields() {
|
|
const fieldNames = this.fields.keys();
|
|
for (const fieldName of fieldNames) {
|
|
this.removeField(fieldName);
|
|
}
|
|
}
|
|
|
|
remove() {
|
|
this.context.database.removeCollection(this.name);
|
|
}
|
|
|
|
async removeFromDb(options?: QueryInterfaceDropTableOptions) {
|
|
if (
|
|
!this.isView() &&
|
|
(await this.existsInDb({
|
|
transaction: options?.transaction,
|
|
}))
|
|
) {
|
|
const queryInterface = this.db.sequelize.getQueryInterface();
|
|
await queryInterface.dropTable(this.getTableNameWithSchema(), options);
|
|
}
|
|
this.remove();
|
|
}
|
|
|
|
async existsInDb(options?: Transactionable) {
|
|
return this.db.queryInterface.collectionTableExists(this, options);
|
|
}
|
|
|
|
removeField(name: string): void | Field {
|
|
if (!this.fields.has(name)) {
|
|
return;
|
|
}
|
|
|
|
const field = this.fields.get(name);
|
|
|
|
const bool = this.fields.delete(name);
|
|
|
|
if (bool) {
|
|
if (this.isParent()) {
|
|
for (const child of this.db.inheritanceMap.getChildren(this.name, {
|
|
deep: false,
|
|
})) {
|
|
const childCollection = this.db.getCollection(child);
|
|
const existField = childCollection.getField(name);
|
|
if (existField && existField.options.inherit) {
|
|
childCollection.removeField(name);
|
|
}
|
|
}
|
|
}
|
|
|
|
this.emit('field.afterRemove', field);
|
|
}
|
|
|
|
return field as Field;
|
|
}
|
|
|
|
/**
|
|
* TODO
|
|
*/
|
|
updateOptions(options: CollectionOptions, mergeOptions?: any) {
|
|
let newOptions = lodash.cloneDeep(options);
|
|
newOptions = merge(this.options, newOptions, mergeOptions);
|
|
this.context.database.emit('beforeUpdateCollection', this, newOptions);
|
|
this.options = newOptions;
|
|
|
|
this.setFields(options.fields, false);
|
|
if (options.repository) {
|
|
this.setRepository(options.repository);
|
|
}
|
|
|
|
this.context.database.emit('afterUpdateCollection', this);
|
|
return this;
|
|
}
|
|
|
|
setSortable(sortable) {
|
|
if (!sortable) {
|
|
return;
|
|
}
|
|
if (sortable === true) {
|
|
this.setField('sort', {
|
|
type: 'sort',
|
|
hidden: true,
|
|
});
|
|
}
|
|
if (typeof sortable === 'string') {
|
|
this.setField(sortable, {
|
|
type: 'sort',
|
|
hidden: true,
|
|
});
|
|
} else if (typeof sortable === 'object') {
|
|
const { name, ...opts } = sortable;
|
|
this.setField(name || 'sort', { type: 'sort', hidden: true, ...opts });
|
|
}
|
|
}
|
|
|
|
/**
|
|
* TODO
|
|
*
|
|
* @param name
|
|
* @param options
|
|
*/
|
|
updateField(name: string, options: FieldOptions) {
|
|
if (!this.hasField(name)) {
|
|
throw new Error(`field ${name} not exists`);
|
|
}
|
|
|
|
if (options.name && options.name !== name) {
|
|
this.removeField(name);
|
|
}
|
|
|
|
this.setField(options.name || name, options);
|
|
}
|
|
|
|
addIndex(index: string | string[] | { fields: string[]; unique?: boolean; [key: string]: any }) {
|
|
if (!index) {
|
|
return;
|
|
}
|
|
|
|
// collection defined indexes
|
|
const indexes: any = this.model.options.indexes || [];
|
|
|
|
let indexName = [];
|
|
let indexItem;
|
|
|
|
if (typeof index === 'string') {
|
|
indexItem = {
|
|
fields: [index],
|
|
};
|
|
indexName = [index];
|
|
} else if (Array.isArray(index)) {
|
|
indexItem = {
|
|
fields: index,
|
|
};
|
|
indexName = index;
|
|
} else if (index?.fields) {
|
|
indexItem = index;
|
|
indexName = index.fields;
|
|
}
|
|
|
|
if (lodash.isEqual(this.model.primaryKeyAttributes, indexName)) {
|
|
return;
|
|
}
|
|
|
|
const name: string = this.model.primaryKeyAttributes.join(',');
|
|
|
|
if (name.startsWith(`${indexName.join(',')},`)) {
|
|
return;
|
|
}
|
|
|
|
for (const item of indexes) {
|
|
if (lodash.isEqual(item.fields, indexName)) {
|
|
return;
|
|
}
|
|
const name: string = item.fields.join(',');
|
|
if (name.startsWith(`${indexName.join(',')},`)) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!indexItem) {
|
|
return;
|
|
}
|
|
|
|
indexes.push(indexItem);
|
|
|
|
const tableName = this.model.getTableName();
|
|
// @ts-ignore
|
|
this.model._indexes = this.model.options.indexes
|
|
// @ts-ignore
|
|
.map((index) => Utils.nameIndex(this.model._conformIndex(index), tableName))
|
|
.map((item) => {
|
|
if (item.name && item.name.length > 63) {
|
|
item.name = 'i_' + md5(item.name);
|
|
}
|
|
return item;
|
|
});
|
|
|
|
this.refreshIndexes();
|
|
}
|
|
|
|
removeIndex(fields: any) {
|
|
if (!fields) {
|
|
return;
|
|
}
|
|
// @ts-ignore
|
|
const indexes: any[] = this.model._indexes;
|
|
// @ts-ignore
|
|
this.model._indexes = indexes.filter((item) => {
|
|
return !lodash.isEqual(item.fields, fields);
|
|
});
|
|
this.refreshIndexes();
|
|
}
|
|
|
|
refreshIndexes() {
|
|
// @ts-ignore
|
|
const indexes: any[] = this.model._indexes;
|
|
|
|
// @ts-ignore
|
|
this.model._indexes = lodash.uniqBy(
|
|
indexes
|
|
.filter((item) => {
|
|
return item.fields.every((field) =>
|
|
Object.values(this.model.rawAttributes).find((fieldVal) => fieldVal.field === field),
|
|
);
|
|
})
|
|
.map((item) => {
|
|
if (this.options.underscored) {
|
|
item.fields = item.fields.map((field) => snakeCase(field));
|
|
}
|
|
return item;
|
|
}),
|
|
'name',
|
|
);
|
|
}
|
|
|
|
async sync(syncOptions?: SyncOptions) {
|
|
const modelNames = new Set([this.model.name]);
|
|
|
|
const { associations } = this.model;
|
|
|
|
for (const associationKey in associations) {
|
|
const association = associations[associationKey];
|
|
modelNames.add(association.target.name);
|
|
|
|
if ((<any>association).through) {
|
|
modelNames.add((<any>association).through.model.name);
|
|
}
|
|
}
|
|
|
|
const models: ModelStatic<Model>[] = [];
|
|
// @ts-ignore
|
|
this.context.database.sequelize.modelManager.forEachModel((model) => {
|
|
if (modelNames.has(model.name)) {
|
|
models.push(model);
|
|
}
|
|
});
|
|
|
|
for (const model of models) {
|
|
await model.sync(syncOptions);
|
|
}
|
|
}
|
|
|
|
public isInherited() {
|
|
return false;
|
|
}
|
|
|
|
public isParent() {
|
|
return this.context.database.inheritanceMap.isParentNode(this.name);
|
|
}
|
|
|
|
public getTableNameWithSchema() {
|
|
const tableName = this.model.tableName;
|
|
|
|
if (this.collectionSchema() && this.db.inDialect('postgres')) {
|
|
return this.db.utils.addSchema(tableName, this.collectionSchema());
|
|
}
|
|
|
|
return tableName;
|
|
}
|
|
|
|
public tableNameAsString(options?: { ignorePublicSchema: boolean }) {
|
|
const tableNameWithSchema = this.getTableNameWithSchema();
|
|
if (lodash.isString(tableNameWithSchema)) {
|
|
return tableNameWithSchema;
|
|
}
|
|
|
|
const schema = tableNameWithSchema.schema;
|
|
const tableName = tableNameWithSchema.tableName;
|
|
|
|
if (options?.ignorePublicSchema && schema === 'public') {
|
|
return tableName;
|
|
}
|
|
|
|
return `${schema}.${tableName}`;
|
|
}
|
|
|
|
public getTableNameWithSchemaAsString() {
|
|
const tableName = this.model.tableName;
|
|
|
|
if (this.collectionSchema() && this.db.inDialect('postgres')) {
|
|
return `${this.collectionSchema()}.${tableName}`;
|
|
}
|
|
|
|
return tableName;
|
|
}
|
|
|
|
public quotedTableName() {
|
|
return this.db.utils.quoteTable(this.getTableNameWithSchema());
|
|
}
|
|
|
|
public collectionSchema() {
|
|
if (this.options.schema) {
|
|
return this.options.schema;
|
|
}
|
|
|
|
if (this.db.options.schema) {
|
|
return this.db.options.schema;
|
|
}
|
|
|
|
if (this.db.inDialect('postgres')) {
|
|
return 'public';
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
public isView() {
|
|
return false;
|
|
}
|
|
|
|
protected sequelizeModelOptions() {
|
|
const { name } = this.options;
|
|
return {
|
|
..._.omit(this.options, ['name', 'fields', 'model', 'targetKey']),
|
|
modelName: name,
|
|
sequelize: this.context.database.sequelize,
|
|
tableName: this.tableName(),
|
|
};
|
|
}
|
|
|
|
private checkOptions(options: CollectionOptions) {
|
|
checkIdentifier(options.name);
|
|
this.checkTableName();
|
|
}
|
|
|
|
private checkTableName() {
|
|
const tableName = this.tableName();
|
|
for (const [k, collection] of this.db.collections) {
|
|
if (
|
|
collection.name != this.options.name &&
|
|
tableName === collection.tableName() &&
|
|
collection.collectionSchema() === this.collectionSchema()
|
|
) {
|
|
throw new Error(`collection ${collection.name} and ${this.name} have same tableName "${tableName}"`);
|
|
}
|
|
}
|
|
}
|
|
|
|
private bindFieldEventListener() {
|
|
this.on('field.afterAdd', (field: Field) => {
|
|
field.bind();
|
|
});
|
|
|
|
this.on('field.afterRemove', (field: Field) => {
|
|
field.unbind();
|
|
this.db.emit('field.afterRemove', field);
|
|
});
|
|
}
|
|
}
|