mirror of
https://github.com/nocobase/nocobase
synced 2024-11-16 03:25:10 +00:00
24ea83f0ff
* create-nocobase-app template from [develop] * change create-nocobase-app package.json config * feat: load configuration from directory * feat: configuration repository toObject * feat: create application from configuration dir * feat: application factory with plugins options * export type * feat: read application config & application with plugins options * feat: release command * fix: database release * chore: workflow package.json * feat: nocobase cli package * feat: console command * chore: load application in command * fix: load packages from process.cwd * feat: cli load env file * feat: create-nocobase-app * fix: gitignore create-nocobase-app lib * fix: sqlite path * feat: create plugin * chore: plugin files template * chore: move cli into application * chore: create-nocobase-app * fix: create plugin * chore: app-client && app-server * chore: package.json * feat: create-nocobase-app download template from npm * chore: create-nocobase-app template * fix: config of plugin-users * fix: yarn.lock * fix: database build error * fix: yarn.lock * fix: resourcer config * chore: cross-env * chore: app-client dependents * fix: env * chore: v0.6.0-alpha.1 * chore: verdaccio * chore(versions): 😊 publish v0.6.0 * chore(versions): 😊 publish v0.6.1-alpha.0 * chore(versions): 😊 publish v0.6.2-alpha.0 * chore(versions): 😊 publish v0.6.2-alpha.1 * chore: 0.6.2-alpha.2 * feat: workspaces * chore(versions): 😊 publish v0.6.2-alpha.3 * chore(versions): 😊 publish v0.6.2-alpha.4 * chore: create-nocobase-app * chore: create-nocobase-app lib * fix: update tsconfig.jest.json * chore: .env * chore(versions): 😊 publish v0.6.2-alpha.5 * chore(versions): 😊 publish v0.6.2-alpha.6 * feat: improve code * chore(versions): 😊 publish v0.6.2-alpha.7 * fix: cleanup * chore(versions): 😊 publish v0.6.2-alpha.8 * chore: tsconfig for app server package * fix: move files * fix: move files Co-authored-by: chenos <chenlinxh@gmail.com>
115 lines
3.3 KiB
TypeScript
115 lines
3.3 KiB
TypeScript
import { Model as SequelizeModel, ModelCtor } from 'sequelize';
|
|
import { Collection } from './collection';
|
|
import { Database } from './database';
|
|
import lodash from 'lodash';
|
|
import { Field } from './fields';
|
|
|
|
interface IModel {
|
|
[key: string]: any;
|
|
}
|
|
|
|
interface JSONTransformerOptions {
|
|
model: ModelCtor<any>;
|
|
collection: Collection;
|
|
db: Database;
|
|
key?: string;
|
|
field?: Field;
|
|
}
|
|
|
|
export class Model<TModelAttributes extends {} = any, TCreationAttributes extends {} = TModelAttributes>
|
|
extends SequelizeModel<TModelAttributes, TCreationAttributes>
|
|
implements IModel
|
|
{
|
|
public static database: Database;
|
|
public static collection: Collection;
|
|
|
|
public toJSON<T extends TModelAttributes>(): T {
|
|
const handleObj = (obj, options: JSONTransformerOptions) => {
|
|
const handles = [
|
|
(data) => {
|
|
if (data instanceof Model) {
|
|
return data.toJSON();
|
|
}
|
|
|
|
return data;
|
|
},
|
|
this.hiddenObjKey,
|
|
];
|
|
return handles.reduce((carry, fn) => fn.apply(this, [carry, options]), obj);
|
|
};
|
|
|
|
const handleArray = (arrayOfObj, options: JSONTransformerOptions) => {
|
|
const handles = [this.sortAssociations];
|
|
return handles.reduce((carry, fn) => fn.apply(this, [carry, options]), arrayOfObj || []);
|
|
};
|
|
|
|
const opts = {
|
|
model: this.constructor as ModelCtor<any>,
|
|
collection: (this.constructor as any).collection,
|
|
db: (this.constructor as any).database as Database,
|
|
};
|
|
|
|
const traverseJSON = (data: T, options: JSONTransformerOptions): T => {
|
|
const { model, db, collection } = options;
|
|
// handle Object
|
|
data = handleObj(data, options);
|
|
|
|
const result = {};
|
|
for (const key of Object.keys(data)) {
|
|
// @ts-ignore
|
|
if (model.hasAlias(key)) {
|
|
const association = model.associations[key];
|
|
const opts = {
|
|
model: association.target,
|
|
collection: db.getCollection(association.target.name),
|
|
db,
|
|
key,
|
|
field: collection.getField(key),
|
|
};
|
|
|
|
if (['HasMany', 'BelongsToMany'].includes(association.associationType)) {
|
|
result[key] = handleArray(data[key], opts).map((item) => traverseJSON(item, opts));
|
|
} else {
|
|
result[key] = data[key] ? traverseJSON(data[key], opts) : null;
|
|
}
|
|
} else {
|
|
result[key] = data[key];
|
|
}
|
|
}
|
|
|
|
return result as T;
|
|
};
|
|
|
|
return traverseJSON(super.toJSON(), opts);
|
|
}
|
|
|
|
private hiddenObjKey(obj, options: JSONTransformerOptions) {
|
|
const hiddenFields = Array.from(options.collection.fields.values())
|
|
.filter((field) => field.options.hidden)
|
|
.map((field) => field.options.name);
|
|
|
|
return lodash.omit(obj, hiddenFields);
|
|
}
|
|
|
|
private sortAssociations(data, { field }: JSONTransformerOptions): any {
|
|
const sortBy = field.options.sortBy;
|
|
return sortBy ? this.sortArray(data, sortBy) : data;
|
|
}
|
|
|
|
private sortArray(data, sortBy: string | string[]) {
|
|
if (!lodash.isArray(sortBy)) {
|
|
sortBy = [sortBy];
|
|
}
|
|
|
|
const orderItems = [];
|
|
const orderDirections = [];
|
|
|
|
sortBy.forEach((sortItem) => {
|
|
orderDirections.push(sortItem.startsWith('-') ? 'desc' : 'asc');
|
|
orderItems.push(sortItem.replace('-', ''));
|
|
});
|
|
|
|
return lodash.orderBy(data, orderItems, orderDirections);
|
|
}
|
|
}
|