Katherine 5d5f455b3c
feat: supports configuring dynamic environment variables and secrets (#5966)
* feat: environments plugin

* feat: improve code

* fix: improve code

* feat: improve code

* refactor: package description

* feat: bulk import

* fix: remove

* refactor: file manager support environment variables

* refactor: file manager support environment variables

* refactor: map manager support environment variables

* refactor: support environment variables

* refactor: support environment variables

* refactor: support delete environment variables

* fix: bug

* refactor: workflow support environment variables

* refactor: email  environment variables

* refactor: support bulk import

* refactor: support bulk import

* refactor: support bulk import

* refactor: support bulk import

* refactor: code improve

* feat: env

* chore: update

* feat: environment

* fix: bug

* fix: acl snippet

* fix: acl snippets

* chore: map manager

* refactor: support line break

* refactor: support password

* chore: environment variables

* fix: bug

* fix: bug

* chore: enviroment variables

* chore: system settings

* fix: improve code

* feat: verification

* feat: map

* feat: file-manager

* feat: notification

* fix: bug

* feat: workflow

* fix: improve code

* fix: bug

* feat: data-source

* feat: auth

* fix: error

* fix: bug

* refactor: description

* refactor: locale

* refactor: locale

* refactor: locale

* refactor: code improve

* refactor: locale

* refactor: locale

* style: style improve

* fix: error

* fix: bug

* fix: bug

* refactor: environment

* fix: ellipsis

* refactor: password

* fix: bug

* fix: bug

* fix: bug

* fix: bug

* fix: bug

* chore: test

* fix: cache

* fix: mysql dialect options

* refactor: email config form

* fix: bug

* fix: bug

* fix: authenticator.dataValues parse

* fix: include undefined

* fix: json

* fix: json parse

* chore: enviromentProvider

* fix: acl

* fix: rowKey

* fix: update ProviderOptions.tsx

* feat: get app instance

* fix: bug

* fix: text

* fix: build error

* fix: error

* chore: migration rules options

* chore: migration rules

* refactor: code improve

* feat: env v2

* chore: environment varibales

* chore: environment serve

* fix: getVariables

* feat: improve code

* fix: bug

* chore: collection options for migration

* chore: tree collection options

* chore: migration rules

* chore: migration rules

* chore: env api

* chore: env api

* fix: optionsKeysNotAllowedInEnv

* fix: required true

* fix: improve code

* fix: app refresh

* fix: remove db.import

* fix: type error

* fix: map

* refactor: locale improve

* refactor: tx-cos

* fix: undefined

* refactor: code improve

* chore: use bookworm

* fix: npm add user

* fix: npm login

* fix: npm adduser

* fix: npm adduser

* fix: expect

* fix: expect

* fix: environmentVariables

* refactor: support bulk delete & filter

* refactor: locale improve

* feat: filter

* refactor: useGlobalVariable

* fix: scope

* fix: bug

* fix: optionsKeysNotAllowedInEnv

* fix: test error

* fix: test

* fix: test

* feat: improve code

---------

Co-authored-by: chenos <chenlinxh@gmail.com>
Co-authored-by: Chareice <chareice@live.com>
2025-01-08 09:32:49 +08:00

170 lines
4.2 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.
*/
import { ACL } from '@nocobase/acl';
import { Logger } from '@nocobase/logger';
import { getNameByParams, parseRequest, ResourceManager } from '@nocobase/resourcer';
import { wrapMiddlewareWithLogging } from '@nocobase/utils';
import EventEmitter from 'events';
import compose from 'koa-compose';
import { DataSourceManager } from './data-source-manager';
import { loadDefaultActions } from './load-default-actions';
import { ICollectionManager } from './types';
export type DataSourceOptions = any;
export type LoadingProgress = {
total: number;
loaded: number;
};
export abstract class DataSource extends EventEmitter {
public collectionManager: ICollectionManager;
public resourceManager: ResourceManager;
public acl: ACL;
public dataSourceManager: DataSourceManager;
logger: Logger;
constructor(protected options: DataSourceOptions) {
super();
this.init(options);
}
_sqlLogger: Logger;
get sqlLogger() {
return this._sqlLogger || this.logger;
}
get name() {
return this.options.name;
}
static testConnection(options?: any): Promise<boolean> {
return Promise.resolve(true);
}
setDataSourceManager(dataSourceManager: DataSourceManager) {
this.dataSourceManager = dataSourceManager;
}
setLogger(logger: Logger) {
this.logger = logger;
}
setSqlLogger(logger: Logger) {
this._sqlLogger = logger;
}
init(options: DataSourceOptions = {}) {
this.acl = this.createACL();
this.resourceManager = this.createResourceManager({
prefix: process.env.API_BASE_PATH,
...options.resourceManager,
});
this.collectionManager = this.createCollectionManager(options);
if (this.collectionManager) {
this.collectionManager.setDataSource(this);
}
this.resourceManager.registerActionHandlers(loadDefaultActions());
if (options.acl !== false) {
this.resourceManager.use(this.acl.middleware(), { tag: 'acl', after: ['auth'] });
}
}
middleware(middlewares: any = []) {
const dataSource = this;
if (!this['_used']) {
for (const [fn, options] of middlewares) {
this.resourceManager.use(fn, options);
}
this['_used'] = true;
}
return async (ctx, next) => {
ctx.dataSource = dataSource;
ctx.getCurrentRepository = () => {
const { resourceName, resourceOf } = ctx.action;
return this.collectionManager.getRepository(resourceName, resourceOf);
};
const middlewares = [this.collectionToResourceMiddleware(), this.resourceManager.middleware()];
return compose(middlewares.map((fn) => wrapMiddlewareWithLogging(fn)))(ctx, next);
};
}
createACL() {
return new ACL();
}
createResourceManager(options) {
return new ResourceManager(options);
}
publicOptions() {
return null;
}
emitLoadingProgress(progress: LoadingProgress) {
this.emit('loadingProgress', progress);
}
async load(options: any = {}) {}
async close() {}
abstract createCollectionManager(options?: any): ICollectionManager;
protected collectionToResourceMiddleware() {
const self = this;
return async function collectionToResource(ctx, next) {
const params = parseRequest(
{
path: ctx.request.path,
method: ctx.request.method,
},
{
prefix: self.resourceManager.options.prefix,
accessors: self.resourceManager.options.accessors,
},
);
if (!params) {
return next();
}
const resourceName = getNameByParams(params);
// 如果资源名称未被定义
if (self.resourceManager.isDefined(resourceName)) {
return next();
}
const splitResult = resourceName.split('.');
const collectionName = splitResult[0];
if (!self.collectionManager.hasCollection(collectionName)) {
return next();
}
self.resourceManager.define({
name: resourceName,
});
return next();
};
}
}