mirror of
https://github.com/nocobase/nocobase
synced 2024-11-17 09:48:00 +00:00
7779cd79ac
* fix: perform load action on boot main app * feat: add dataType option in collection duplicator * chore: reset optional dumpable config * chore: dump command * chore: dump & restore command * chore: delay restore * fix: dump test * chore: restore command * chore: dump command action * chore: dumpable collection api * chore: client collection option * feat: backup& restore client * chore: content disposition header in dump response * chore: download backup field * feat: collection origin option * fix: test * chore: collection manager collection origin * chore: upload backup field * chore: upload restore file * chore: upload restore file * fix: test * chore: backup and restore support learn more * refactor: upload restore file * refactor: upload restore file * fix: test * fix: test * chore: dumpable collection with title * chore: pg only test * chore: test * fix: test * chore: test sleep * style: locale improve * refactor: download backup file * refactor: start restore * fix: restore key name * refactor: start restore * refactor: start restore * refactor: start restore * refactor: start restore * refactor: start restore * refactor: start restore * chore: unify duplicator option * fix: dump empty collection * chore: test * chore: test * style: style improve * refactor: locale improve * chore: dumpalbe collection orders * style: style improve * style: style improve * style: icon adjust * chore: nginx body size * chore: get file status * feat: run dump task * feat: download api * chore: backup files resourcer * feat: restore destroy api * chore: backup files resoucer * feat: list backup files action * chore: get collection meta from dumped file * fix: dump file name * fix: test * chore: backup and restore ui * chore: swagger api for backup & restore * chore: api doc * chore: api doc * chore: api doc * chore: backup and restore ui * chore: backup and restore ui * chore: backup and restore ui * chore: backup and restore ui * chore: backup and restore ui * fix: restore values * style: style improve * fix: download field respontype * fix: restore form local file * refactor: local improve * refactor: delete backup file * fix: in progress status * refactor: locale improve * refactor: locale improve * refactor: style improve * refactor: style improve * refactor: style improve * test: dump collection table attribute * chore: dump collection with table attributes * chore: test * chore: create new table in restore * fix: import error * chore: restore table from backup file * chore: sync collection after restore collections * fix: restore json data * style: style improve * chore: restore with fields * chore: test * fix: test * fix: test with underscored * style: style improve * fix: lock file state * chore: add test file * refactor: backup & restore plugin * fix: mysql test * chore: skip import view collection * chore: restore collection with inherits topo order * fix: import * style: style improve * fix: restore sequence fields * fix: themeConfig collection duplicator option * fix: restore with dialectOnly meta * fix: throw error * fix: restore * fix: import backup file created in postgres into mysql * fix: repeated items in inherits * chore: upgrade after restore * feat: check database env before restore * feat: handle autoincr val in postgres * chore: sqlite & mysql queryInterface * chore: test * fix: test * chore: test * fix: build * fix: pg test * fix: restore with date field * chore: theme-config collection * chore: chage import collections method to support collection origin * chore: fallback get autoincr value in mysql * fix: dataType normalize * chore: delay restore * chore: test * fix: build * feat: collectin onDump * feat: collection onDump interface * chore: dump with view collection * chore: sync in restore * refactor: locale improve * refactor: code improve * fix: test * fix: data sync * chore: rename backup & restore plugin * chore: skip test * style: style improve * style: style improve * style: style improve * style: style improve * chore: import version check * chore: backup file dir * chore: build * fix: bugs * fix: error * fix: pageSize * fix: import origin * fix: improve code * fix: remove namespace * chore: dump rules config * fix: dump custom collection * chore: version * fix: test * fix: test * fix: test * fix: test * chore: test * fix: load custom collection * fix: client * fix: translation * chore: code * fix: bug * fix: support shared option * fix: roles collection dumpRules * chore: test * fix: define collections * chore: collection group * fix: translation * fix: translation * fix: restore options * chore: restore command * refactor: optimize the command line * chore: dump error * fix: test error * fix: test error * fix: test error * fix: test error * fix: test error * fix: skip cli test cases * fix: test error * fix: too many open files * fix: update migration version * fix: migrations * fix: upgrade * fix: error * fix: migration error * fix: upgrade * fix: test error * fix: timeout * fix: width * feat: auto load collections * fix: test error * fix: test error * fix: test error * fix: test error * fix: test error * fix: test error * fix: test error * fix: ipc error * fix: test error --------- Co-authored-by: Chareice <chareice@live.com> Co-authored-by: katherinehhh <katherine_15995@163.com>
236 lines
5.2 KiB
TypeScript
236 lines
5.2 KiB
TypeScript
import { Model } from '@nocobase/database';
|
|
import { LoggerOptions } from '@nocobase/logger';
|
|
import { fsExists, importModule } from '@nocobase/utils';
|
|
import fs from 'fs';
|
|
import glob from 'glob';
|
|
import type { TFuncKey, TOptions } from 'i18next';
|
|
import { basename, resolve } from 'path';
|
|
import { Application } from './application';
|
|
import { InstallOptions, getExposeChangelogUrl, getExposeReadmeUrl } from './plugin-manager';
|
|
import { checkAndGetCompatible } from './plugin-manager/utils';
|
|
|
|
export interface PluginInterface {
|
|
beforeLoad?: () => void;
|
|
|
|
load();
|
|
|
|
getName(): string;
|
|
}
|
|
|
|
export interface PluginOptions {
|
|
activate?: boolean;
|
|
displayName?: string;
|
|
description?: string;
|
|
version?: string;
|
|
enabled?: boolean;
|
|
install?: (this: Plugin) => void;
|
|
load?: (this: Plugin) => void;
|
|
plugin?: typeof Plugin;
|
|
|
|
[key: string]: any;
|
|
}
|
|
|
|
export abstract class Plugin<O = any> implements PluginInterface {
|
|
options: any;
|
|
app: Application;
|
|
model: Model;
|
|
state: any = {};
|
|
|
|
constructor(app: Application, options?: any) {
|
|
this.app = app;
|
|
this.setOptions(options);
|
|
}
|
|
|
|
get log() {
|
|
return this.app.log.child({
|
|
reqId: this.app.context.reqId,
|
|
module: this.name,
|
|
});
|
|
}
|
|
|
|
get name() {
|
|
return this.options.name as string;
|
|
}
|
|
|
|
get pm() {
|
|
return this.app.pm;
|
|
}
|
|
|
|
get db() {
|
|
return this.app.db;
|
|
}
|
|
|
|
get enabled() {
|
|
return this.options.enabled;
|
|
}
|
|
|
|
set enabled(value) {
|
|
this.options.enabled = value;
|
|
}
|
|
|
|
get installed() {
|
|
return this.options.installed;
|
|
}
|
|
|
|
set installed(value) {
|
|
this.options.installed = value;
|
|
}
|
|
|
|
get isPreset() {
|
|
return this.options.isPreset;
|
|
}
|
|
|
|
setOptions(options: any) {
|
|
this.options = options || {};
|
|
}
|
|
|
|
getName() {
|
|
return (this.options as any).name;
|
|
}
|
|
|
|
createLogger(options: LoggerOptions) {
|
|
return this.app.createLogger(options);
|
|
}
|
|
|
|
get _sourceDir() {
|
|
if (basename(__dirname) === 'src') {
|
|
return 'src';
|
|
}
|
|
return this.isPreset ? 'lib' : 'dist';
|
|
}
|
|
|
|
async loadCommands() {
|
|
const extensions = ['js', 'ts'];
|
|
const directory = resolve(
|
|
process.env.NODE_MODULES_PATH,
|
|
this.options.packageName,
|
|
this._sourceDir,
|
|
'server/commands',
|
|
);
|
|
const patten = `${directory}/*.{${extensions.join(',')}}`;
|
|
const files = glob.sync(patten, {
|
|
ignore: ['**/*.d.ts'],
|
|
});
|
|
for (const file of files) {
|
|
let filename = basename(file);
|
|
filename = filename.substring(0, filename.lastIndexOf('.')) || filename;
|
|
const callback = await importModule(file);
|
|
callback(this.app);
|
|
}
|
|
if (files.length) {
|
|
this.app.log.debug(`load commands [${this.name}]`);
|
|
}
|
|
}
|
|
|
|
async loadMigrations() {
|
|
this.app.log.debug(`load plugin migrations [${this.name}]`);
|
|
if (!this.options.packageName) {
|
|
return { beforeLoad: [], afterSync: [], afterLoad: [] };
|
|
}
|
|
const directory = resolve(
|
|
process.env.NODE_MODULES_PATH,
|
|
this.options.packageName,
|
|
this._sourceDir,
|
|
'server/migrations',
|
|
);
|
|
return await this.app.loadMigrations({
|
|
directory,
|
|
namespace: this.options.packageName,
|
|
context: {
|
|
plugin: this,
|
|
},
|
|
});
|
|
}
|
|
|
|
afterAdd() {}
|
|
|
|
beforeLoad() {}
|
|
|
|
async load() {}
|
|
|
|
async install(options?: InstallOptions) {}
|
|
|
|
async upgrade() {}
|
|
|
|
async beforeEnable() {}
|
|
|
|
async afterEnable() {}
|
|
|
|
async beforeDisable() {}
|
|
|
|
async afterDisable() {}
|
|
|
|
async beforeRemove() {}
|
|
|
|
async afterRemove() {}
|
|
|
|
async importCollections(collectionsPath: string) {
|
|
// await this.db.import({
|
|
// directory: collectionsPath,
|
|
// from: `plugin:${this.getName()}`,
|
|
// });
|
|
}
|
|
|
|
async loadCollections() {
|
|
if (!this.options.packageName) {
|
|
return;
|
|
}
|
|
const directory = resolve(
|
|
process.env.NODE_MODULES_PATH,
|
|
this.options.packageName,
|
|
this._sourceDir,
|
|
'server/collections',
|
|
);
|
|
if (await fsExists(directory)) {
|
|
await this.db.import({
|
|
directory,
|
|
from: this.options.packageName,
|
|
});
|
|
}
|
|
}
|
|
|
|
requiredPlugins() {
|
|
return [];
|
|
}
|
|
|
|
t(text: TFuncKey | TFuncKey[], options: TOptions = {}) {
|
|
return this.app.i18n.t(text, { ns: this.options['packageName'], ...(options as any) });
|
|
}
|
|
|
|
async toJSON(options: any = {}) {
|
|
const { locale = 'en-US' } = options;
|
|
const { name, packageName, packageJson } = this.options;
|
|
if (!packageName) {
|
|
return {
|
|
...this.options,
|
|
};
|
|
}
|
|
|
|
const results = {
|
|
...this.options,
|
|
readmeUrl: getExposeReadmeUrl(packageName, locale),
|
|
changelogUrl: getExposeChangelogUrl(packageName),
|
|
displayName: packageJson[`displayName.${locale}`] || packageJson.displayName || name,
|
|
description: packageJson[`description.${locale}`] || packageJson.description,
|
|
};
|
|
|
|
if (!options.withOutOpenFile) {
|
|
const file = await fs.promises.realpath(
|
|
resolve(process.env.NODE_MODULES_PATH || resolve(process.cwd(), 'node_modules'), packageName),
|
|
);
|
|
|
|
return {
|
|
...results,
|
|
...(await checkAndGetCompatible(packageName)),
|
|
lastUpdated: (await fs.promises.stat(file)).ctime,
|
|
file,
|
|
updatable: file.startsWith(process.env.PLUGIN_STORAGE_PATH),
|
|
};
|
|
}
|
|
|
|
return results;
|
|
}
|
|
}
|
|
|
|
export default Plugin;
|